我需要修改一个现有的程序,它包含以下代码:
var inputs = events.Select(async ev => await ProcessEventAsync(ev))
.Select(t => t.Result)
.Where(i => i != null)
.ToList();
但这对我来说很奇怪,首先使用async和await select。根据Stephen Cleary的回答,我应该可以放弃这些。
然后是第二个Select,它选择结果。这是否意味着任务根本不是异步的,而是同步执行的(如此多的努力是徒劳的),或者任务将异步执行,当它完成时,执行查询的其余部分?
根据Stephen Cleary的另一个回答,我应该像下面这样写上面的代码吗?
var tasks = await Task.WhenAll(events.Select(ev => ProcessEventAsync(ev)));
var inputs = tasks.Where(result => result != null).ToList();
这是完全一样的吗?
var inputs = (await Task.WhenAll(events.Select(ev => ProcessEventAsync(ev))))
.Where(result => result != null).ToList();
当我在这个项目上工作时,我想改变第一个代码示例,但我不太热衷于改变(显然工作)异步代码。也许我只是担心什么,所有3个代码样本做完全相同的事情?
processevensasync看起来像这样:
async Task<InputResult> ProcessEventAsync(InputEvent ev) {...}
var inputs = events.Select(async ev => await ProcessEventAsync(ev))
.Select(t => t.Result)
.Where(i => i != null)
.ToList();
但这对我来说很奇怪,首先在选择中使用async和await。根据Stephen Cleary的回答,我应该可以放弃这些。
调用Select是有效的。这两行基本上是相同的:
events.Select(async ev => await ProcessEventAsync(ev))
events.Select(ev => ProcessEventAsync(ev))
(关于如何从proces76async抛出同步异常有一个微小的区别,但在这段代码的上下文中,这一点都不重要。)
然后是第二个Select,它选择结果。这是否意味着任务根本不是异步的,而是同步执行的(如此多的努力是徒劳的),或者任务将异步执行,当它完成时,执行查询的其余部分?
这意味着查询正在阻塞。所以这并不是真正的异步。
分解一下:
var inputs = events.Select(async ev => await ProcessEventAsync(ev))
将首先为每个事件启动异步操作。然后这一行:
.Select(t => t.Result)
将等待这些操作一次完成一个(首先等待第一个事件的操作,然后是下一个,然后是下一个,等等)。
这是我不关心的部分,因为它会阻塞并将任何异常包装在AggregateException中。
这是完全一样的吗?
var tasks = await Task.WhenAll(events.Select(ev => ProcessEventAsync(ev)));
var inputs = tasks.Where(result => result != null).ToList();
var inputs = (await Task.WhenAll(events.Select(ev => ProcessEventAsync(ev))))
.Where(result => result != null).ToList();
是的,这两个例子是等价的。它们都启动所有异步操作(events.Select(…)),然后异步地等待所有操作以任意顺序完成(await Task.WhenAll(…)),然后继续其余的工作(Where…)。
这两个示例都与原始代码不同。原始代码是阻塞的,将在AggregateException中包装异常。
var inputs = events.Select(async ev => await ProcessEventAsync(ev))
.Select(t => t.Result)
.Where(i => i != null)
.ToList();
但这对我来说很奇怪,首先在选择中使用async和await。根据Stephen Cleary的回答,我应该可以放弃这些。
调用Select是有效的。这两行基本上是相同的:
events.Select(async ev => await ProcessEventAsync(ev))
events.Select(ev => ProcessEventAsync(ev))
(关于如何从proces76async抛出同步异常有一个微小的区别,但在这段代码的上下文中,这一点都不重要。)
然后是第二个Select,它选择结果。这是否意味着任务根本不是异步的,而是同步执行的(如此多的努力是徒劳的),或者任务将异步执行,当它完成时,执行查询的其余部分?
这意味着查询正在阻塞。所以这并不是真正的异步。
分解一下:
var inputs = events.Select(async ev => await ProcessEventAsync(ev))
将首先为每个事件启动异步操作。然后这一行:
.Select(t => t.Result)
将等待这些操作一次完成一个(首先等待第一个事件的操作,然后是下一个,然后是下一个,等等)。
这是我不关心的部分,因为它会阻塞并将任何异常包装在AggregateException中。
这是完全一样的吗?
var tasks = await Task.WhenAll(events.Select(ev => ProcessEventAsync(ev)));
var inputs = tasks.Where(result => result != null).ToList();
var inputs = (await Task.WhenAll(events.Select(ev => ProcessEventAsync(ev))))
.Where(result => result != null).ToList();
是的,这两个例子是等价的。它们都启动所有异步操作(events.Select(…)),然后异步地等待所有操作以任意顺序完成(await Task.WhenAll(…)),然后继续其余的工作(Where…)。
这两个示例都与原始代码不同。原始代码是阻塞的,将在AggregateException中包装异常。
我有同样的问题@KTCheek,因为我需要它按顺序执行。然而,我想我会尝试使用IAsyncEnumerable(在。net Core 3中引入)和await foreach(在c# 8中引入)。下面是我想出的:
public static class IEnumerableExtensions {
public static async IAsyncEnumerable<TResult> SelectAsync<TSource, TResult>(this IEnumerable<TSource> source, Func<TSource, Task<TResult>> selector) {
foreach (var item in source) {
yield return await selector(item);
}
}
}
public static class IAsyncEnumerableExtensions {
public static async Task<List<TSource>> ToListAsync<TSource>(this IAsyncEnumerable<TSource> source) {
var list = new List<TSource>();
await foreach (var item in source) {
list.Add(item);
}
return list;
}
}
可以这样解释:
var inputs = await events.SelectAsync(ev => ProcessEventAsync(ev)).ToListAsync();
更新:或者你可以添加一个对System.Linq.Async的引用,然后你可以说:
var inputs = await events
.ToAsyncEnumerable()
.SelectAwait(async ev => await ProcessEventAsync(ev))
.ToListAsync();