如何使用.NET Action执行参数数量未知的方法?

时间:2012-11-06 22:18:09

标签: c# multithreading lambda action

我想在向用户显示进度条的同时对工作线程执行某些操作。我创建了一个类

public class ProgressBar
{
   public void StartAsyncTask(Action action)
   {
      Task t = new Task(action);
      t.start();
   }
}

我发现我可以通过以下方式向StartAsyncTask发送任何方法:

  ProgressBar pb = new ProgressBar();
  pb.StartAsyncTask( () => DoSomething(15, "something"));

  public void DoSomething(int i, string s)
   {
      //do something
   }

首先,我似乎无法理解lambda表达式是什么以及如何 - () => - 已翻译以及Action对象如何传递具有未知数量参数的委托。

我想在我的ProgressBar中使用BackgroundWorker,但在这种情况下,我需要调用该操作。所以像这样:

void m_backgroundWorker_DoWork(object sender, DoWorkEventArgs e)
{
    Action action = e.Argument as Action; //same action here passed through EventArgs
   //but already in a worker thread so no need for the Task object 

   //and now i need to somehow invoke the action object but i don't know what the parameters are. 

   action.Invoke( ? ); 
}

在第一个示例中如何在不知道StartAsyncTask(Action action)方法中的参数的情况下执行操作?

在这种情况下,为什么我需要知道调用动作时的参数?

关于如何/为何/何时使用“Action”的所有内容对我来说都很不清楚,即使我在这里阅读了MSDN文档和其他一些帖子。 任何有关这方面的信息都会对我有帮助。

2 个答案:

答案 0 :(得分:50)

我认为你有点过分思考。所以让我们从顶部开始:

  1. lambda表达式是引用方法执行的表示法。例如:

     x => x + 3
    

    在最基本的级别,这表示一个函数,它接受1个输入x,然后返回一个等于x + 3的值。所以在你的情况下,你的表达:

    () => DoSomething(15, "Something")
    

    表示采用0参数的方法,然后调用方法DoSomething(15, "Something")。编译器在幕后将其转换为FuncAction委托给您。所以它实际上是:

    new Action(delegate()
    {
        DoSomething(15, "Something")
    }); 
    

    上面简单表达式的编译器重写将是:

    new Func<int, int>(delegate(int x)
    {
        return x + 3;
    });
    
  2. 接下来,如果您想稍后调用某个操作,那么这样做的语法非常简单:

    Action someMethod = new Action(() => { Console.WriteLine("hello world"); }));
    someMethod(); // Invokes the delegate
    

    因此,如果您有一个给定的Action实例,只需使用()语法调用它就可以了,因为Action是一个接受0参数并且不返回任何内容的委托。

    功能同样容易:

    Func<int, int> previousGuy = x => x + 3;
    var result = previousGuy(3); // result is 6
    
  3. 最后,如果您想要传递一个方法来调用,并且此时没有参数的上下文,您可以简单地将您的调用包装在一个操作中,然后再调用它。例如:

    var myAction = new Action(() =>
         {
              // Some Complex Logic
              DoSomething(15, "Something");
              // More Complex Logic, etc
         });
    
    InvokeLater(myAction);
    
    public void InvokeLater(Action action)
    {
          action();
    }
    

    所有数据都在方法的闭包中捕获,因此被保存。因此,如果您可以设法使用Action属性将e.Argument传递给您的活动,那么您需要做的就是致电(e.Argument as Action)()

答案 1 :(得分:2)

您是否可以在该委托上使用DynamicInvoke()(以params object[] args作为参数)

action.DynamicInvoke(arg1, arg2, arg3 );