当发生异常时,如何取消iAsyncencentoser?
我正在使用iasyncenumerable
来获取大量数据 - 像这样:
static async IAsyncEnumerable<int> GetSequence([EnumeratorCancellation] CancellationToken ct = default)
{
for (int i = 1; i <= 10; i++)
{
var data = await DoSomethingAsync() // get data from web request
yield return data;
}
}
现在是我的问题:如果dosomethingsasync
thr thr thr tragriation(也许是网络问题),我想取消剩余的请求,有可能吗?
我正在尝试以下代码来处理,但仍然需要帮助:
CancellationTokenSource source = new CancellationTokenSource();
CancellationToken token = source.Token;
try
{
IAsyncEnumerable<int> dataCol = GetSequence(token);
await foreach (var d in dataCol)
{
// do ...
}
}
catch (Exception)
{
source.Cancel(true);
throw;
}
感谢某人有最佳实践
I am using IAsyncEnumerable
to fetch a lot of data - like this:
static async IAsyncEnumerable<int> GetSequence([EnumeratorCancellation] CancellationToken ct = default)
{
for (int i = 1; i <= 10; i++)
{
var data = await DoSomethingAsync() // get data from web request
yield return data;
}
}
Now here is my question: if DoSomethingAsync
throws an exception (maybe a network issue), I would like to cancel the remaining requests, is it possible?
I am trying below code to handle but still need help:
CancellationTokenSource source = new CancellationTokenSource();
CancellationToken token = source.Token;
try
{
IAsyncEnumerable<int> dataCol = GetSequence(token);
await foreach (var d in dataCol)
{
// do ...
}
}
catch (Exception)
{
source.Cancel(true);
throw;
}
Appreciate if someone has best practices
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
data:image/s3,"s3://crabby-images/d5906/d59060df4059a6cc364216c4d63ceec29ef7fe66" alt="扫码二维码加入Web技术交流群"
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(1)
来自原始
等同于;
调用
movenextAsync
导致您的get sequence
方法执行到下一个收益率返回
,或者直到抛出未手动的异常为止。执行该方法不会恢复,直到下一个呼叫moveNextAsync
。get sequence
和等待着foreach
不同行运行,他们轮流执行更多的代码在同步的任务中。,则
等待e.movenextasync()
将重新添加异常。然后将处置枚举者。然后,异常将堆栈流到下一个异常处理程序。您无需弄乱取消令牌即可防止
获取序列
恢复执行。实际上,从中分解出
循环也将中止执行,好像收益率返回
从未恢复过。在get sequence
方法中,导致任何终于{}
块执行,如果有的话。From the original language proposal
is equivalent to;
Calling
MoveNextAsync
causes yourGetSequence
method to execute up to the nextyield return
, or until an unhandled exception is thrown. Execution of that method will not resume until the next call toMoveNextAsync
.GetSequence
andawait foreach
do not run in parallel, they take turns executing more code within the same async task.then
await e.MoveNextAsync()
will re-throw the exception. Then enumerator will be disposed. Then the exception would flow up the stack to the next exception handler.You don't need to mess around with cancelation tokens to prevent
GetSequence
from resuming execution. In fact breaking out of theawait foreach
loop would also abort execution, as if theyield return
never resumed. Causing anyfinally{}
blocks in yourGetSequence
method to execute, if you had any.