网站建设心得体会500字,西安网站建设 招聘,wordpress模板推荐,网站开发工程师培训在异步函数中#xff0c;一般使用CancellationToken来控制函数的执行。这个Token需要作为参数传递到异步函数中#xff1a;
public staic TaskT DoAsync(CancellationToken token)
{...
} 那么如果一个异步函数没有这个Token参数#xff0c;如何取消呢? 之前看到一… 在异步函数中一般使用CancellationToken来控制函数的执行。这个Token需要作为参数传递到异步函数中
public staic TaskT DoAsync(CancellationToken token)
{...
} 那么如果一个异步函数没有这个Token参数如何取消呢? 之前看到一个博客How to Cancel a Non-Cancellable Task in C# 1.构建取消异步函数的扩展方法 先上代码 public static class TaskExtensions{public static async TaskT WithCancellationT (this TaskT task,CancellationToken cancellationToken){var tcsnew TaskCompletionSourceobject(TaskCreationOptions.RunContinuationsAsynchronously);using (cancellationToken.Register(state {((TaskCompletionSourceobject)state!).TrySetResult(null!);}, tcs)){var resultTask await Task.WhenAny(task, tcs.Task);if(resultTasktcs.Task){throw new OperationCanceledException(cancellationToken);}return await task;}}} 显然这是一个扩展方法旨在为TaskT类型扩展一个名叫WithCancellation函数。这个函数会有一个CancellationToken,但这个Token不是传递给任务参数task(也就是我们要取消的函数)的而是用于外部控制。 代码中首先构建了一个异步任务利用TaskCompletionSourceT可以构建一个用于控制原任务的异步任务这里的TaskCompletionSource不多解释了可以参考博客here。然后通过传入的Token注册一个回调函数回调函数传入的参数就是刚刚创立的TaskCompletionSourceT对象回调函数会调用成员函数TrySetResult()给任务赋值。而回调函数的执行则是在Token被取消时触发。
2. 测试 然后我们使用这个扩展方法构建实例 internal class CancelTaskWithoutCancellationToken{private static readonly CancellationTokenSource cancellationTokenSource new CancellationTokenSource();public static async Task Test(){try{var result await Task.Run(async () {await Task.Delay(TimeSpan.FromSeconds(5));Console.WriteLine(操作完成);await Task.Delay(300);Console.WriteLine(还继续吗);return 7;}).WithCancellation(cancellationTokenSource.Token);Console.WriteLine(result.ToString());}catch (Exception ex){Console.WriteLine(ex.Message);Console.WriteLine(任务取消了);}finally{cancellationTokenSource.Dispose();}}public static void Cancel(){cancellationTokenSource?.Cancel();}}
测试代码
var t1CancelTaskWithoutCancellationToken.Test();
var t2 Task.Run(()
{Thread.Sleep(1000);CancelTaskWithoutCancellationToken.Cancel();
});
await Task.WhenAll(t1, t2);
运行结果如下 好像结果很符合预期。
假设测试代码后面还有一些任务要运行也就是主线程没那么快结束呢让我们在测试代码后面加一行
var t1CancelTaskWithoutCancellationToken.Test();
var t2 Task.Run(()
{Thread.Sleep(1000);CancelTaskWithoutCancellationToken.Cancel();
});
await Task.WhenAll(t1, t2);
Console.WriteLine(----------------------);
Console.ReadLine();
运行结果 Oh No原来的任务还是执行了说明没能阻止那个任务继续运行所以原博客说取消一个不能被取消的任务non-cancelable是错的开工没有回头箭。但是从前面的例子我们可以取消等待那个任务。 实际上博主在另外一篇博客找到了关于这个问题的说明 How do I cancel non-cancelable async operations? 博主在最后总结 So, can you cancel non-cancelable operations? No. Can you cancel waits on non-cancelable operations? Sure… just be very careful when you do. 所以如果想能随时取消一个Task,最稳妥的办法还是将Token作为参数传递进去