繁体   English   中英

等待事件从任务列表中触发

[英]Awaiting events to be fired from task list

我正在使用Task.Run()启动几个任务。 在每个任务中,我都为InstallCompleted定义了一个事件处理程序。 然后,我await这些任务完成。 棘手的是,下面看到的installModule.Install()方法会立即返回,因为它在内部调用了OneWay WCF服务调用。

那么,如何await直到上述每个任务中的installModule所有实例都触发了各自的InstallCompleted事件? 如果有更好的方法可以做到这一点,我全神贯注。

var installTasks = new List<Task>();

foreach (var targetMachine in vm.TargetMachines)
{
    var installTask = Task.Run(() =>
    {
        foreach (var module in Modules)
        {
            var installModule = module as IInstallModule;  // copy closure variable
            if (installModule == null)
                continue;

            installModule.InstallCompleted += (s, e) =>
            {
                //TODO set some flag indicating this install is complete
            };

            var ip = new Progress<InstallProgress>();
            installModule.Install(targetMachine.MachineName, ip);
        }
    });

    installTasks.Add(installTask);
}

await Task.WhenAll(installTasks);  // unblocks immediately because installModule.Install() above returns immediately

每当你互操作async与另一种异步API的 ,你可以使用TaskCompletionSource<T>作为一个异步一个时间信号。 就个人而言,我喜欢将此作为扩展方法:

public static Task<InstallCompletedResult> InstallAsync(
    this IInstallModule module, string machineName,
    IProgress<InstallProgress> progress = null)
{
  var tcs = new TaskCompletionSource<InstallCompletedResult>();
  InstallCompletedEventHandler handler = null;
  handler = (s, e) =>
  {
    module.InstallCompleted -= handler;
    if (e.Exception != null)
      tcs.TrySetException(e.Exception);
    else
      tcs.TrySetResult(e.Result);
  };
  module.InstallCompleted += handler;
  module.Install(machineName, progress);
  return tcs.Task;
}

一旦有了互操作性,就可以很容易地进行消费:

var installTask = Task.Run(async () =>
{
  var tasks = new List<Task>();
  foreach (var module in Modules)
  {
    var installModule = module as IInstallModule;  // copy closure variable
    if (installModule == null)
      continue;

    var ip = new Progress<InstallProgress>();
    tasks.Add(installModule.InstallAsync(targetMachine.MachineName, ip));
  }
});

或者,更简单地说:

var installTask = Task.Run(async () =>
{
  var tasks = Modules.OfType<IInstallModule>()
      .Select(module => module.InstallAsync(targetMachine.MachineName,
          new Progress<InstallProgress>()));
  await Task.WhenAll(tasks);
});

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM