为什么不能将匿名方法作为参数传递给BeginInvoke
方法?我有以下代码:
private delegate void CfgMnMnuDlg(DIServer svr);
private void ConfigureMainMenu(DIServer server,)
{
MenuStrip mnMnu = PresenterView.MainMenu;
if (mnMnu.InvokeRequired)
{
mnMnu.BeginInvoke((CfgMnMnuDlg)ConfigureMainMenu,
new object[] { server});
}
else
{
// Do actual work here
}
}
我试图避免声明代表。为什么我不能写下面的东西呢?或者我可以,我只是无法弄清楚正确的语法?以下目前生成:
参数类型'Anonymous method'不能分配给参数类型'System.Delegate'
好的,这当然是正确的,但是我可以使用其他一些语法来做到这一点(避免为了使用BeginInvoke()
而声明一个单独的委托吗?
(能够做到这一点完全符合使用anon方法/ lamdas代替显式委托的概念,这些委托在其他地方干净利落地工作。)
private void ConfigureMainMenu(DIServer server,)
{
MenuStrip mnMnu = PresenterView.MainMenu;
if (mnMnu.InvokeRequired)
{
mnMnu.BeginInvoke( // pass anonymous method instead ?
delegate(DIServer svr) { ConfigureMainMenu(server);},
new object[] { server});
}
else
{
// Do actual work here
}
}
答案 0 :(得分:36)
试试这个:
control.BeginInvoke((MethodInvoker) delegate { /* method details */ });
或者:
private void ConfigureMainMenu(DIServer server)
{
if (control.InvokeRequired)
{
control.BeginInvoke(new Action<DIServer >(ConfigureMainMenu), server);
}
else
{
/* do work */
}
}
或者:
private void ConfigureMainMenu(DIServer server)
{
MenuStrip mnMnu = PresenterView.MainMenu;
if (mnMnu.InvokeRequired)
{
// Private variable
_methodInvoker = new MethodInvoker((Action)(() => ConfigureMainMenu(server)));
_methodInvoker.BeginInvoke(new AsyncCallback(ProcessEnded), null); // Call _methodInvoker.EndInvoke in ProcessEnded
}
else
{
/* do work */
}
}
答案 1 :(得分:2)
你应该可以这样写:
private void ConfigureMainMenu(DIServer server,)
{
MenuStrip mnMnu = PresenterView.MainMenu;
if (mnMnu.InvokeRequired)
{
mnMnu.BeginInvoke(new Action<DIServer>(ConfigureMainMenu),
new object[] { server});
}
else
{
// Do actual work here
}
}
答案 2 :(得分:1)
您可以编写一个扩展方法来包装匿名方法,甚至可以处理InvokeRequired
语义:
public static void InvokeAction(this Control ctl, Action a)
{
if (!ctl.InvokeRequired)
{
a();
}
else
{
ctl.BeginInvoke(new MethodInvoker(a));
}
}
这将允许你这样做:
control.InvokeAction(delegate() { ConfigureMainMenu(server); });
答案 3 :(得分:1)
您可以通过调用自己调用在单个方法中执行此操作:
ClassData updData = new ClassData();
this.BeginInvoke(new Action<ClassData>(FillCurve),
new object[] { updData });
...
public void FillCurve(ClassData updData)
{
...
}
答案 4 :(得分:0)
我尝试了很多不同的方法,但都没有用。即...
// Fails -- cannot convert lamda to System.Delegate
mnMnu.BeginInvoke( (DIServer svr)=> {ConfigureMainMenu(server);}, new object[] server);
// Fails -- cannot convert anonymous method to System.Delegate
mnMnu.BeginInvoke( new delegate(DIServer svr){ConfigureMainMenu(server);}, new object[] server);
所以,简短的回答是否定的。你可以在给定的上下文中创建简短的助手委托,并使用lambdas使它更整洁但是就是这样。
编辑:事实证明我错了。下面的methodinvoker答案有效。 见page答案 5 :(得分:0)
对于具有有限数量参数的完全匿名方法:
Func<int, int?> caller = new Func<int, int?>((int param1) =>
{
return null;
});
caller.BeginInvoke(7, new AsyncCallback((IAsyncResult ar) =>
{
AsyncResult result = (AsyncResult)ar;
Func<int, int?> action = (Func<int, int?>)result.AsyncDelegate;
action.EndInvoke(ar);
}), null);
您可以根据需要使用其他Func委托类型之一。