c# - 如何从 .NET 4.5 中的并行任务中产生

标签 c# task-parallel-library async-await c#-5.0 yield-return

我想将 .NET 迭代器与并行任务/等待一起使用?像这样:

IEnumerable<TDst> Foo<TSrc, TDest>(IEnumerable<TSrc> source)
{
    Parallel.ForEach(
        source,
        s=>
        {
            // Ordering is NOT important
            // items can be yielded as soon as they are done                
            yield return ExecuteOrDownloadSomething(s);
        }
}

不幸的是,.NET 本身无法处理这个问题。 @svick 迄今为止的最佳答案 - 使用 AsParallel()。

奖励:任何实现多个发布者和单个订阅者的简单异步/等待代码?订户会屈服,而酒吧会处理。 (仅限核心库)

最佳答案

这似乎是 PLINQ 的工作:

return source.AsParallel().Select(s => ExecuteOrDownloadSomething(s));

这将使用有限数量的线程并行执行委托(delegate),并在完成后立即返回每个结果。

如果ExecuteOrDownloadSomething()方法是 IO 绑定(bind)的(例如它实际上下载了一些东西)并且你不想浪费线程,然后使用 async - await可能有道理,但会更复杂。

如果你想充分利用async ,你不应该返回 IEnumerable ,因为它是同步的(即,如果没有可用项目,它会阻塞)。你需要的是某种异步收集,你可以使用 ISourceBlock (特别是 TransformBlock )来自 TPL 数据流:

ISourceBlock<TDst> Foo<TSrc, TDest>(IEnumerable<TSrc> source)
{
    var block = new TransformBlock<TSrc, TDest>(
        async s => await ExecuteOrDownloadSomethingAsync(s),
        new ExecutionDataflowBlockOptions
        {
            MaxDegreeOfParallelism = DataflowBlockOptions.Unbounded
        });

    foreach (var item in source)
        block.Post(item);

    block.Complete();

    return block;
}

如果源“慢”(即您希望在迭代 Foo() 完成之前开始处理来自 source 的结果),您可能需要移动 foreachComplete()调用一个单独的 Task .更好的解决方案是制作 source进入ISourceBlock<TSrc>也是。

关于c# - 如何从 .NET 4.5 中的并行任务中产生,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14806240/

相关文章:

c# - httpclient async/await 与否

node.js - 如何使用 aws-sdk 将文件同步上传到 S3?

c# - 如何在 c# Xamarin 中实现 ios 委托(delegate)/数据源模式?

c# - Entity Framework 查询缓存

c# - 从数据库中检索数据的最快方法

c# - 在 C# 中使用自定义 Linux 驱动程序

c# - TPL 的性能计数器?

c# - TaskCancellationException 如何避免成功控制流的异常?

c# - 异步任务的 NUnit 测试在 C# 中失败

c# - 如何在异步方法的单元测试中测试 ConfigureAwait(false)?