0

0

c# CancellationToken 和 IAsyncDisposable 的结合使用

畫卷琴夢

畫卷琴夢

发布时间:2026-01-19 12:29:40

|

697人浏览过

|

来源于php中文网

原创

CancellationToken 不能直接取消 IAsyncDisposable.DisposeAsync(),因为其签名固定为 ValueTask DisposeAsync(),不接收 CancellationToken 参数;需手动保存并检查 token,在清理逻辑中显式传递或响应。

c# cancellationtoken 和 iasyncdisposable 的结合使用

为什么 CancellationToken 不能直接取消 IAsyncDisposable.DisposeAsync()

因为 IAsyncDisposable.DisposeAsync() 本身不接收 CancellationToken 参数——它的签名固定为 ValueTask DisposeAsync()。你无法像 HttpClient.GetAsync(uri, cancellationToken) 那样传入 token 来中断释放逻辑。这意味着:如果 DisposeAsync() 内部执行了耗时的异步清理(比如等待远端服务确认断连、刷盘、关闭长连接),它不会响应外部取消请求。

在 DisposeAsync 中手动检查 CancellationToken 的正确姿势

必须把 CancellationToken 持有为字段或闭包变量,并在 DisposeAsync() 实现里显式轮询或传递给可取消的子操作。常见错误是只在构造时捕获 token,却没在释放路径中使用它。

  • CancellationToken 存为私有只读字段(如 private readonly CancellationToken _ct;),构造时传入并保存
  • DisposeAsync() 中优先检查 _ct.IsCancellationRequested,快速返回
  • _ct 传给所有支持 cancel 的子调用,例如 stream.WriteAsync(buffer, _ct)semaphore.WaitAsync(_ct)
  • 避免在 DisposeAsync() 中启动新 Task.Run 或忽略 token 的阻塞调用(如 Thread.Sleep、无 token 版 Wait()
public class ResilientResource : IAsyncDisposable
{
    private readonly CancellationTokenSource _cts;
    private readonly Stream _stream;

    public ResilientResource(Stream stream, CancellationToken ct = default)
    {
        _stream = stream;
        _cts = CancellationTokenSource.CreateLinkedTokenSource(ct);
    }

    public async ValueTask DisposeAsync()
    {
        if (_cts.Token.IsCancellationRequested)
            return;

        try
        {
            await _stream.FlushAsync(_cts.Token).ConfigureAwait(false);
            await _stream.DisposeAsync().ConfigureAwait(false);
        }
        catch (OperationCanceledException) when (_cts.Token.IsCancellationRequested)
        {
            // 预期行为:子操作因 token 被取消,静默处理
        }
        finally
        {
            _cts.Dispose();
        }
    }
}

结合 using await 和 CancellationToken 的实际调用方式

调用方必须确保 CancellationToken 在整个生命周期内有效,且不能在 using await 块外提前触发 Cancel(),否则 DisposeAsync() 可能刚进入就立即退出,导致资源未清理。

酷炫粒子碎片视频播放特效
酷炫粒子碎片视频播放特效

three基于canvas全屏酷炫的粒子碎片化和mp4视频文件结合播放特效。支持全屏mp4视频播放查看效果。只需修改mp4视频文件即可替换使用。ps:本地需要localhost跨域才能演示

下载
  • 不要用 var cts = new CancellationTokenSource(); cts.Cancel(); using await ... —— token 已失效
  • 推荐在 async 方法内统一管理:用 using var cts = new CancellationTokenSource(timeoutMs);,再传入构造函数和后续逻辑
  • using await 语句本身不接受 token,取消只能靠资源内部响应,所以依赖上面的手动检查机制
  • 若需“强制超时取消 DisposeAsync”,得在外层套一层 WaitAsync(ct) 包装,但要注意这会丢失原生 ValueTask 性能优势
await using var resource = new ResilientResource(stream, cts.Token);
await DoWorkAsync(resource, cts.Token);

// 此处 cts.Cancel() 触发后,resource.DisposeAsync() 内部会响应

容易被忽略的线程安全与状态竞争点

DisposeAsync() 可能被多次调用(尤其在异常路径或并发场景下),而 CancellationToken 的检查和清理逻辑若没加锁或状态标记,会导致重复释放、NullReference 或竞态取消。

  • 添加私有 private volatile bool _disposed; 字段,首次进入时设为 true,后续直接 return
  • 不要在 DisposeAsync() 中修改共享状态后再检查 IsCancellationRequested —— 顺序错乱会导致漏响应
  • CancellationToken.Register() 是危险操作:注册的回调可能在 token 已取消后仍执行,且难以控制执行上下文;优先用 ThrowIfCancellationRequested()WaitAsync(ct)
  • 如果底层资源(如 NetworkStream)本身不支持 cancel on dispose,强行注入 token 也无效——得换更底层可控的实现
CancellationToken 和 IAsyncDisposable 的结合不是自动生效的魔法,关键在于你是否在 DisposeAsync 的每一行清理代码里,都把它当作一个需要主动倾听的信号。

相关专题

更多
登录token无效
登录token无效

登录token无效解决方法:1、检查token的有效期限,如果token已经过期,需要重新获取一个新的token;2、检查token的签名,如果签名不正确,需要重新获取一个新的token;3、检查密钥的正确性,如果密钥不正确,需要重新获取一个新的token;4、使用HTTPS协议传输token,建议使用HTTPS协议进行传输 ;5、使用双因素认证,双因素认证可以提高账户的安全性。

6097

2023.09.14

登录token无效怎么办
登录token无效怎么办

登录token无效的解决办法有检查Token是否过期、检查Token是否正确、检查Token是否被篡改、检查Token是否与用户匹配、清除缓存或Cookie、检查网络连接和服务器状态、重新登录或请求新的Token、联系技术支持或开发人员等。本专题为大家提供token相关的文章、下载、课程内容,供大家免费下载体验。

808

2023.09.14

token怎么获取
token怎么获取

获取token值的方法:1、小程序调用“wx.login()”获取 临时登录凭证code,并回传到开发者服务器;2、开发者服务器以code换取,用户唯一标识openid和会话密钥“session_key”。想了解更详细的内容,可以阅读本专题下面的文章。

1062

2023.12.21

token什么意思
token什么意思

token是一种用于表示用户权限、记录交易信息、支付虚拟货币的数字货币。可以用来在特定的网络上进行交易,用来购买或出售特定的虚拟货币,也可以用来支付特定的服务费用。想了解更多token什么意思的相关内容可以访问本专题下面的文章。

1259

2024.03.01

c++中volatile关键字的作用
c++中volatile关键字的作用

本专题整合了c++中volatile关键字的相关内容,阅读专题下面的文章了解更多详细内容。

69

2025.10.23

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

481

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

61

2025.12.01

go语言闭包相关教程大全
go语言闭包相关教程大全

本专题整合了go语言闭包相关数据,阅读专题下面的文章了解更多相关内容。

135

2025.07.29

PS使用蒙版相关教程
PS使用蒙版相关教程

本专题整合了ps使用蒙版相关教程,阅读专题下面的文章了解更多详细内容。

23

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号