Вопрос

Почему вы не можете передать анонимный метод в качестве параметра 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
    }
}

Я пытаюсь избежать объявления делегата.Почему я не могу вместо этого написать что-то вроде приведенного ниже?Или я могу, и я просто не могу понять правильный синтаксис?Ниже в настоящее время генерируется:

Тип аргумента «Анонимный метод» нельзя назначить типу параметра «System.Delegate».

Хорошо, это, конечно, правильно, но есть ли какой-нибудь другой синтаксис, который я могу использовать для этого (избегайте объявления отдельного делегата, чтобы использовать BeginInvoke()?

(Возможность сделать это прекрасно вписывается в концепцию использования анонимных методов/лямбд вместо явных делегатов, которые так хорошо работают везде.)

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
    }
}
Это было полезно?

Решение

Попробуй это:

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 */
    }
}

Другие советы

Вы должны быть в состоянии написать что-то вроде этого:

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
    }
}

Вы можете написать метод расширения, который обернет анонимные методы и даже позаботится о семантике 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); });

Вы можете сделать это одним методом, вызвав вызов самого себя:

  ClassData updData =  new ClassData();

  this.BeginInvoke(new Action<ClassData>(FillCurve),
                           new object[] { updData });

...

public void FillCurve(ClassData updData)
{
 ...
}

Для полностью анонимных методов с ограниченным количеством параметров:

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.

Я пробовал несколько разных методов, но ни один из них не работает. то есть ...


// 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);

Итак, короткий ответ - нет. Вы можете создавать короткие вспомогательные делегаты в данном контексте и использовать лямбда-выражения, чтобы сделать его немного более аккуратным, но это в значительной степени так.

РЕДАКТИРОВАТЬ: Оказывается, я не прав. Метод invoker ответ ниже работает. Посетите эту страницу

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top