Silverlight/WPF中,如果要在多线程中对界面控件值做修改,用Dispatcher对象的BeginInvoke方法无疑是最方便的办法 ,见:温故而知新:WinForm/Silverlight多线程编程中如何更新UI控件的值
但今天发现WPF中的BeginInvoke却无法自动将匿名方法/Lambda表达式转变成Delegate类型(注:对委托,匿名方法,Lambda感到陌生的朋友先阅读温故而知新:Delegate,Action,Func,匿名方法,匿名委托,事件)
silverlight中的代码片段:
private void button1_Click(object sender, RoutedEventArgs e) { Thread t = new Thread(TestMethod); t.Start(); Thread t2 = new Thread(TestMethod2); t2.Start("Hello World"); } void TestMethod() { this.Dispatcher.BeginInvoke(() => { this.textBlock1.Text = DateTime.Now.ToString("HH:mm:ss"); }); } void TestMethod2(object s) { this.Dispatcher.BeginInvoke(() => { this.textBlock1.Text =s.ToString() ; }); }
WPF中如果这样用,会报如下错误:
Cannot convert lambda expression to type 'System.Delegate' because it is not a delegate type
即:无法将lambda表达式转换为"System.Delegate",因为它不是delegate 类型
即使把Lambda表达式改成匿名方法的写法也不行:
public void TestMethod() { this.Dispatcher.BeginInvoke(delegate() { this.textBlock1.Text = DateTime.Now.ToString("HH:mm:ss fff"); }); }
仍然会报错:
Cannot convert anonymous method to type 'System.Delegate' because it is not a delegate type
即:无法将匿名方法转换为"System.Delegate",因为它不是delegate 类型
当然也可以自己定义一个Delegate类型用最传统的方法来写:
delegate void MyDelegate(); delegate void MyDelegate2(object s); public void TestMethod() { MyDelegate d = new MyDelegate(UpdateText); this.Dispatcher.BeginInvoke(d); } void UpdateText() { this.textBlock1.Text = DateTime.Now.ToString("HH:mm:ss fff"); } void UpdateText2(object s) { this.textBlock1.Text = s.ToString(); } public void TestMethod2(object s) { MyDelegate2 d = new MyDelegate2(UpdateText2); this.Dispatcher.BeginInvoke(d, "Hello World"); }
但是这种写法太繁琐了,还得单独把方法的定义提取出来,同时还要定义相应的委托类型,难道不能象Silverlght中那样清爽一点么?
既然出错的原因就是编译器不自动做类型转换,那我们就来强制转换吧
public void TestMethod() { this.Dispatcher.BeginInvoke((Action)delegate() { this.textBlock1.Text = DateTime.Now.ToString("HH:mm:ss fff"); }); } public void TestMethod2(object s) { this.Dispatcher.BeginInvoke((Action)(() => { this.textBlock1.Text = s.ToString(); })); }
这样就可以了,把匿名方法/Lambda表达式强制转换为Action,而Action实质就是委托类型,so,问题解决了!
不过仍然有点疑问:为啥编译器能自动认别Silverlight,却不认WPF呢?这算不算是编译器的BUG(或是需要改进的地方)