14

我试图理解为什么 BeginInvoke 方法不接受匿名方法。

void bgWorker_ProgressChanged(object sender, ProgressChangedEventArgs e)
{
    if (InvokeRequired)
    {
        //Won't compile
        BeginInvoke(delegate(object sender, ProgressChangedEventArgs e) 
        { bgWorker_ProgressChanged(sender, e); });
    }

    progressBar1.Increment(e.ProgressPercentage);
}

它告诉我'无法从'匿名方法'转换为'System.Delegate',而当我将匿名方法转换为委托时它确实有效?

BeginInvoke((progressDelegate)delegate { bgWorker_ProgressChanged(sender, e); });
4

3 回答 3

16

您需要告诉编译器要创建什么类型的委托,因为Invoke(等)只是采用Delegate(而不是更具体的东西)。

适用于最大的受众,MethodInvoker是一种方便的代表类型

BeginInvoke((MethodInvoker) delegate(...) {...});

但是......BackgroundWorker.ProgressChanged自动在 UI 线程上触发 - 所以你甚至不需要这个。

于 2009-06-09T08:11:04.273 回答
7

Delegate 类是委托类型的基类。但是,只有系统和编译器可以显式地从 Delegate 类或 MulticastDelegate 类派生。也不允许从委托类型派生新类型。Delegate 类不被视为委托类型;它是一个用于派生委托类型的类。来源——MSDN

因此需要显式强制转换为派生自委托类型。当您为 System.Delegate 类型的参数传递匿名方法时,您会遇到这个特定的编译器错误——幸运的是,这种情况很少见。这太灵活了。

delegate void MyDelegate();

  static void DoSomething_Flexible(Delegate d)
  {   d.DynamicInvoke();      }
  static void DoSomething_Usable(MyDelegate d)
  {   d();      }
  static void Main(string[] args)
  {
     // requires explicit cast else compile error Error "Cannot convert anonymous method to type 'System.Delegate' because it is not a delegate type    
     DoSomething_Flexible((MyDelegate) delegate { Console.WriteLine("Flexible is here!"); });  

     // Parameter Type is a .NET Delegate, no explicit cast needed here. 
     DoSomething_Usable(delegate { Console.WriteLine("Usable is here!"); });
  }

伊恩·格里菲斯(Ian Griffith)在此页面上对此进行了更多介绍。(见注释标题后的段落)

于 2009-06-09T07:57:30.603 回答
5

在这些情况下,大多数时候您都在处理无参数委托或谓词。对此进行排序的最简单方法是将匿名方法直接转换为任一ActionPredicate分别;您只是不需要为这样的简单事情创建自定义委托类型。

所以你会有类似的东西

BeginInvoke((Action)delegate(){YourCode.DoSomething();});

或者

BeginInvoke((Predicate)delegate(object yourParameter){return YourCode.IsTheParameterSomething(yourParameter)});

高温高压

于 2009-12-08T16:16:49.957 回答