取消异步任务时 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 拥有线稿上色优化、图片重绘、人物姿势检测、涂鸦完善等功能
C#怎么使用CancellationToken C#如何用取消令牌优雅地取消异步任务和长时间操作【进阶】