C#怎么使用CancellationToken C#如何用取消令牌优雅地取消异步任务和长时间操作【进阶】

张开发
2026/4/14 22:50:19 15 分钟阅读

分享文章

C#怎么使用CancellationToken C#如何用取消令牌优雅地取消异步任务和长时间操作【进阶】
取消异步任务时 CancellationToken 不生效的主因是未将同一令牌实例传入所有可取消的异步操作同步代码需手动轮询 IsCancellationRequested 且须在合理位置如循环体内检查。取消异步任务时 CancellationToken 不生效检查是否传入了正确实例最常见的情况是你调用了 Cancel()但目标方法毫无反应。根本原因往往是「没把令牌传进真正干活的函数里」——比如只在方法签名里声明了 CancellationToken token却在内部调用 Task.Delay(1000) 时忘了传它。正确做法是所有可取消的异步原语如 Task.Delay、HttpClient.GetAsync、Stream.ReadAsync都必须显式接收并使用同一个 CancellationToken 实例await Task.Delay(5000, token); // ? 传入await httpClient.GetAsync(https://api.example.com, token); // ? 传入await stream.ReadAsync(buffer, token); // ? 传入别自己写轮子去轮询 token.IsCancellationRequested除非底层 API 不支持 token多个 await 调用之间要共用同一个 token不能每个都 new 一个 CancellationTokenSource().Token如果封装了自定义异步方法务必把 CancellationToken 作为参数透传到底层调用链长时间同步操作比如密集计算或文件读取怎么响应取消同步代码不会自动响应 CancellationToken必须手动轮询。但轮询位置很关键不能只在循环开头检查否则一次迭代耗时太久就失去响应性。典型场景是处理大数组或逐块读文件for (int i 0; i data.Length; i){ token.ThrowIfCancellationRequested(); // ? 每次迭代都检查 Process(data[i]);}不要只在循环外检查一次那等于没取消逻辑避免在阻塞 I/O如 FileStream.Read中轮询——应改用支持 token 的异步版本ReadAsync若必须用同步 I/O可在每次读块后加 token.ThrowIfCancellationRequested()但要注意线程上下文和超时精度CancellationTokenSource 的生命周期管理容易出什么问题忘记调用 Dispose() 或过早释放 CancellationTokenSource会导致资源泄漏或取消信号丢失。常见错误现象OperationCanceledException 没抛出、后续调用突然失败、GC 压力异常升高。 稿定AI 拥有线稿上色优化、图片重绘、人物姿势检测、涂鸦完善等功能

更多文章