有的可以直接用“lambda 表达式”作为嘱托参数,但有的却不行,为什么呢
有的可以直接用“lambda 表达式”作为委托参数,但有的却不行,为什么呢?
前两个例子是可以的,但后一个例子就不行,大家看看是为什么,谢谢!!!
可简写成:
下面是Func委托的定义
public delegate TResult Func<in T, out TResult>(T arg);
再比如:
我也可以简写成:
这是ThreadStart委托的定义
public delegate void ThreadStart();
都可以直接用“lambda 表达式”作为委托参数,但下面的却不行:
在多线程中更新用户界面时,需要调用Invoke方法来更新用户界面,比如:
我想简写成下面的样子:
却出现错误错误:
无法将 lambda 表达式 转换为类型“System.Delegate”,因为它不是委托类型
但若写成下面的,就没问题了:
label1.Invoke((UpdateUIDelegate)(n => { label1.Text = n.ToString(); }), i);
Invoke的参数如下:
public object Invoke(Delegate method, params object[] args);
和前面不同的是这里的参数是Delegate
我查阅了一些资料,说delegate定义的委托派生自MulticastDelegate,而MulticastDelegate又派生自Delegate,Delegate又派生自object.
我们都知道:
object obj=25;//不需要强制类型转换
int i=(int)object;//需要强制类型转换
Invoke的参数是Delegate,是其他委托的基类,为何这里还需要强制类型转换呢?
------解决思路----------------------
因为 Delegate 和 MulticastDelegate 本身都是 abstract 的,实际使用时必须有一个具体的委托继承自它们才能new。前两个例子里面方法的参数都是具体的委托类型,所以编译器可以支持自动new那个类型的委托,就能够直接写lambda。而Invoke要求的参数是 Delegate,你给lambda编译器没法知道要转换成哪一个具体类型的委托,就没法自动帮忙new了,所以必须自己指定好具体的委托类型。一般调用Invoke都是 .Invoke(new Action(() => ...)); 这样用Action这个具体的委托来封装lambda。
------解决思路----------------------
也可以用强制转换,让编译器有机会推导出具体类型。
label1.Invoke((Action<int>)(n => { label1.Text = n.ToString(); }), i);
前两个例子是可以的,但后一个例子就不行,大家看看是为什么,谢谢!!!
static void Main(string[] args)
{
int[] a = { 8, 2, 6, 56, 46, 18 };
Func<int, bool> d = i => i > 18; //Func委托,返回所给整数参数的值是否大于18
bool b = a.Any(d);//数组中有大于18的数据吗?
Console.WriteLine(b);
}
可简写成:
static void Main(string[] args)
{
int[] a = { 8, 2, 6, 56, 46, 18 };
bool b = a.Any(i => i > 18); // Any是个“扩展方法”
Console.WriteLine(b);
}
下面是Func委托的定义
public delegate TResult Func<in T, out TResult>(T arg);
再比如:
class Program
{
static void Main(string[] args)
{
ThreadStart ts=new ThreadStart(Run);
Thread t = new Thread(ts);
t.Start();
}
static void Run()
{
for (int i = 0; i < 1000; ++i)
{
Console.WriteLine(i);
}
}
}
我也可以简写成:
static void Main(string[] args)
{
Thread t = new Thread(() =>
{
for (int i = 0; i < 1000; ++i)
{
Console.WriteLine(i);
}
});
t.Start();
}
这是ThreadStart委托的定义
public delegate void ThreadStart();
都可以直接用“lambda 表达式”作为委托参数,但下面的却不行:
在多线程中更新用户界面时,需要调用Invoke方法来更新用户界面,比如:
public partial class Form1 : Form
{
private delegate void UpdateUIDelegate(int i);
public Form1()
{
InitializeComponent();
}
private void button1_Click(object sender, EventArgs e)
{
Thread t=new Thread(doWorker);
t.Start();
}
private void UpdateUI(int i)
{
label1.Text = i.ToString();
}
private void doWorker()
{
UpdateUIDelegate ui = UpdateUI;
for (int i = 0; i < 1000; ++i)
{
label1.Invoke(ui, i);
Thread.Sleep(10);
}
}
}
我想简写成下面的样子:
public partial class Form1 : Form
{
private delegate void UpdateUIDelegate(int i);
public Form1()
{
InitializeComponent();
}
private void button1_Click(object sender, EventArgs e)
{
Thread t=new Thread(doWorker);
t.Start();
}
//private void UpdateUI(int i)
//{
// label1.Text = i.ToString();
//}
private void doWorker()
{
for (int i = 0; i < 1000; ++i)
{
label1.Invoke(n => { label1.Text = n.ToString(); }, i); //***出现错误***
Thread.Sleep(10);
}
}
}
却出现错误错误:
无法将 lambda 表达式 转换为类型“System.Delegate”,因为它不是委托类型
但若写成下面的,就没问题了:
label1.Invoke((UpdateUIDelegate)(n => { label1.Text = n.ToString(); }), i);
Invoke的参数如下:
public object Invoke(Delegate method, params object[] args);
和前面不同的是这里的参数是Delegate
我查阅了一些资料,说delegate定义的委托派生自MulticastDelegate,而MulticastDelegate又派生自Delegate,Delegate又派生自object.
我们都知道:
object obj=25;//不需要强制类型转换
int i=(int)object;//需要强制类型转换
Invoke的参数是Delegate,是其他委托的基类,为何这里还需要强制类型转换呢?
------解决思路----------------------
因为 Delegate 和 MulticastDelegate 本身都是 abstract 的,实际使用时必须有一个具体的委托继承自它们才能new。前两个例子里面方法的参数都是具体的委托类型,所以编译器可以支持自动new那个类型的委托,就能够直接写lambda。而Invoke要求的参数是 Delegate,你给lambda编译器没法知道要转换成哪一个具体类型的委托,就没法自动帮忙new了,所以必须自己指定好具体的委托类型。一般调用Invoke都是 .Invoke(new Action(() => ...)); 这样用Action这个具体的委托来封装lambda。
------解决思路----------------------
也可以用强制转换,让编译器有机会推导出具体类型。
label1.Invoke((Action<int>)(n => { label1.Text = n.ToString(); }), i);