- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有两个使用 SemaphoreSlim 和字符串数组“Contents”的循环
一个foreach循环:
var allTasks = new List<Task>();
var throttle = new SemaphoreSlim(10,10);
foreach (string s in Contents)
{
await throttle.WaitAsync();
allTasks.Add(
Task.Run(async () =>
{
try
{
rootResponse.Add(await POSTAsync(s, siteurl, src, target));
}
finally
{
throttle.Release();
}
}));
}
await Task.WhenAll(allTasks);
一个for循环:
var allTasks = new List<Task>();
var throttle = new SemaphoreSlim(10,10);
for(int s=0;s<Contents.Count;s++)
{
await throttle.WaitAsync();
allTasks.Add(
Task.Run(async () =>
{
try
{
rootResponse[s] = await POSTAsync(Contents[s], siteurl, src, target);
}
finally
{
throttle.Release();
}
}));
}
await Task.WhenAll(allTasks);
第一个 foreach 循环运行良好,但 for 循环 Task.WhenAll(allTasks) 返回 OutOfRangeException,我希望 Contents[] 索引和 List 索引匹配。
我可以修复 for 循环吗?还是有更好的方法?
最佳答案
这将解决您当前的问题
for (int s = 0; s < Contents.Count; s++)
{
var content = Contents[s];
allTasks.Add(
Task.Run(async () =>
{
await throttle.WaitAsync();
try
{
rootResponse[s] = await POSTAsync(content, siteurl, src, target);
}
finally
{
throttle.Release();
}
}));
}
await Task.WhenAll(allTasks);
然而,这是一段相当困惑和讨厌的代码。这看起来有点整洁
public static async Task DoStuffAsync(Content[] contents, string siteurl, string src, string target)
{
var throttle = new SemaphoreSlim(10, 10);
// local method
async Task<(Content, SomeResponse)> PostAsyncWrapper(Content content)
{
await throttle.WaitAsync();
try
{
// return a content and result pair
return (content, await PostAsync(content, siteurl, src, target));
}
finally
{
throttle.Release();
}
}
var results = await Task.WhenAll(contents.Select(PostAsyncWrapper));
// do stuff with your results pairs here
}
还有许多其他方法可以做到这一点,PLinq
、Parallel.For
、Parallel.ForEach
,或者只是整理您的捕获在你上面的循环中。
然而,由于您有一个 IO 绑定(bind)工作负载,并且您有运行它的 async
方法。最合适的解决方案是 async
await
模式,Parallel.For
、Parallel.ForEach
都无法满足最佳需求。
另一种方式是TPL DataFlow可以在 System.Threading.Tasks.Dataflow
nuget 中找到的库包。
代码
public static async Task DoStuffAsync(Content[] contents, string siteurl, string src, string target)
{
async Task<(Content, SomeResponse)> PostAsyncWrapper(Content content)
{
return (content, await PostAsync(content, siteurl, src, target));
}
var bufferblock = new BufferBlock<(Content, SomeResponse)>();
var actionBlock = new TransformBlock<Content, (Content, SomeResponse)>(
content => PostAsyncWrapper(content),
new ExecutionDataflowBlockOptions
{
EnsureOrdered = false,
MaxDegreeOfParallelism = 100,
SingleProducerConstrained = true
});
actionBlock.LinkTo(bufferblock);
foreach (var content in contents)
actionBlock.Post(content);
actionBlock.Complete();
await actionBlock.Completion;
if (bufferblock.TryReceiveAll(out var result))
{
// do stuff with your results pairs here
}
}
基本上这会创建一个 BufferBlock
和 TransformBlock
,您将工作负载注入(inject) TransformBlock
,它的选项具有并行度,并将它们插入 BufferBlock
,您等待完成并获得结果。
为什么选择数据流?因为它处理 async
await
,它具有 MaxDegreeOfParallelism
,它专为 IO bound 或 CPU Bound 工作负载而设计,并且使用起来非常简单。此外,由于大多数数据通常以多种方式(在管道中)进行处理,因此您可以使用它按顺序和并行方式或以您选择的任何方式来管道和操作数据流。
祝你好运
关于c# - Task.Whenall(List<Task>) 输出 OutOfRangeException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52911847/
我正在审查一些代码并试图找出您应该或不应该使用 Task.WhenAll(Tasks[]) 的技术原因。本质上是并行进行 Http 调用。 Http 调用调用不同的微服务,我猜其中一个调用可能需要一些
有时我想并行等待一堆任务,但任务本身取决于某些条件。我会写 var tasks = new List { DoThisAsync(), DoThatAsync() }; if (condition)
我有一个方法可以生成一系列任务( See my previous post about threading ),在这个方法结束时,我有以下选项: await Task.WhenAll(tasks);
当前使用以下代码等待任务集合完成。但是,我现在遇到这样一种情况,我希望能够通过取消 token 取消/中止WhenAll调用。我该怎么办? Dim TaskCollection As New Li
每次运行以下程序都会捕获 Mike 的异常。 WhenAll 是顺序涉及每个任务之间的连续上下文还是所有任务都同时运行?如果它是并发的,为什么总是捕获 Mike 的异常而不是 Mitch 的异常。我推
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎与 help center 中定义的范围内的编程无关。 . 关闭 7 年前。 Improve
我需要让 UI 线程等待任务数组完成执行。下面代码的问题是 - 任务反过来调用 UI 线程写入文本框。如何解决这个问题? public partial class FormConsole : Form
考虑以下理想代码(它不起作用)。我实际上是在尝试创建 Task 的列表返回特定对象的 s,将它们与 string 相关联标识符,然后使用 Task.WhenAll 批量执行所有这些.在执行结束时,我需
我有一个类的两个实例,它们创建一个 UDP 套接字以从 UDP 客户端接收数据。如果其中一个实例抛出异常,我想立即在更高层处理它。在我的程序中,它们以 await Task.WhenAll(recv1
这个问题在这里已经有了答案: An async/await example that causes a deadlock (5 个回答) 4年前关闭。 我是 C# 任务的新手,遇到了一个我不理解的问题
我有两个异步方法作为单独的线程/任务在窗体窗口的后台运行。这些是无限循环,只是在后台做一些工作,然后使用调度程序更新 UI。见下文。 public async Task RunCameraTh
我有一个简单的 Winforms 应用程序。我想后台 TCP 连接/打印请求,并在我的代码中的设定点检查所有任务的输出。 我希望 ReportOnTasks阻止直到 WaitAll完成了。请有人能解释
我对异步编程还很陌生。我想在等待其中任何一个任务之前启动一系列任务(发出 http 请求)。 List identifiers; //Set identifiers to what they shou
我有一些关于 TPL 的问题。正如您所看到的,我正在创建 2 个简单的任务并将它们添加到列表中。问题(预期行为)是任务在 WorkMethodAsync 中点击“await”后立即返回,因此使 Tas
我正在尝试理解 async/await 并想知道这是否是 Task.WhenAll 的正确使用方式方法: public class AsyncLib { public async Task>
我有一堆任务定义为: Task t1 = new Task( () => { /* Do Something */ } ); Task t2 = new Task( () => { /* Do Som
我经常有一些代码,其中我有一组独立任务,这些任务创建了可以立即启动的其他独立任务的集合,所以就时间而言,它看起来像下面这样。 A0 ------- B0 --------- C0
跟随 this question (及其答案)我想使用 TaskCompletionSource 和 Task.WhenAll 等待任何任务首先返回 True 。所以我写了这个: TaskComple
我有以下两种方法,它们产生相同的结果。 public static async Task> ExecuteSqlStoredProcedureSelect(IEnumerable linkedAcco
有没有非阻塞的Task.WaitAll类似于 Task.WhenAll ,但不是平行的? 这是我写的,但也许它是内置的? public async Task> AwaitAllAsync(IEnume
我是一名优秀的程序员,十分优秀!