0

0

c# 如何在 IDisposable 对象被并发使用时保证只 Dispose 一次

畫卷琴夢

畫卷琴夢

发布时间:2026-01-14 12:06:02

|

926人浏览过

|

来源于php中文网

原创

IDisposable.Dispose() 并发调用不安全,因多数实现未加线程保护,可能导致重复释放、ObjectDisposedException 或内存损坏;应使用 Interlocked.CompareExchange 原子标记已释放状态,确保 DisposeCore 仅执行一次。

c# 如何在 idisposable 对象被并发使用时保证只 dispose 一次

为什么 IDisposable.Dispose() 并发调用会出问题

很多实现 IDisposable 的类(比如 FileStream、自定义资源包装器)内部没有对 Dispose() 做线程安全防护。多次并发调用 Dispose() 可能导致:重复释放非托管句柄、ObjectDisposedException 被抛出、甚至内存损坏(尤其涉及 SafeHandle 或 P/Invoke 场景)。.NET 本身不保证 Dispose() 是可重入的——它只承诺「调用一次后对象进入已释放状态」,没说「调用多次是否安全」。

Interlocked.CompareExchange 实现原子标记

最轻量、无锁、且被 .NET 运行时广泛采用的方式是用一个 int 字段做“是否已释放”标记,配合 Interlocked.CompareExchange 判断并设置。这是微软StreamTimer 等 BCL 类型中的实际做法。

private int _disposed = 0; // 0 = not disposed, 1 = disposed

public void Dispose()
{
    if (Interlocked.CompareExchange(ref _disposed, 1, 0) == 0)
    {
        // 真正的释放逻辑,只执行一次
        DisposeCore();
        GC.SuppressFinalize(this);
    }
}

private void DisposeCore()
{
    // 释放托管资源(如其他 IDisposable 对象)
    _stream?.Dispose();
    // 释放非托管资源(如 CloseHandle、free())
    if (_handle != IntPtr.Zero)
    {
        NativeMethods.CloseHandle(_handle);
        _handle = IntPtr.Zero;
    }
}

注意 Dispose(bool) 模式下的并发陷阱

如果你沿用经典的双参数 Dispose(bool disposing) 模式,**不能直接在两个入口(Dispose() 和终结器)里都加 Interlocked 判断**——因为终结器线程和用户线程可能同时闯入,而 GC.SuppressFinalize(this) 必须在首次 Dispose() 时就调用,否则终结器仍可能运行。

小鸽子助手
小鸽子助手

一款集成于WPS/Word的智能写作插件

下载
  • Dispose() 方法里必须调用 Interlocked.CompareExchange + GC.SuppressFinalize
  • ~MyClass() 终结器里**只能调用 DisposeCore(false),且不能做任何 Interlocked 检查或再调用 GC.SuppressFinalize**(此时已无意义)
  • 所有资源释放逻辑(包括托管和非托管)统一收口到 DisposeCore(bool disposing),但要根据 disposing 参数决定是否释放托管资源

别依赖 lockMonitor 做 Dispose 同步

看似简单,但风险很高:

  • 如果 Dispose() 内部释放的资源本身涉及同步(比如关闭一个正在被读写的 NetworkStream),再套一层 lock 容易引发死锁
  • 终结器线程不能获取普通锁(Monitor.Enter 在终结器中可能永久阻塞)
  • 性能上,Interlocked 是无锁原子操作,比锁快一个数量级,且无上下文切换开销
  • 只要确保 _disposed 字段是 volatile 或通过 Interlocked 访问,就不需要额外 volatile 声明
真正难的是判断哪些资源允许重复释放、哪些绝对不行。比如 CancellationTokenSource.Cancel() 是幂等的,但 SafeHandle.SetHandleAsInvalid() 不是——一旦设为无效,再次调用会抛异常。所以「只 Dispose 一次」不是为了代码好看,而是防止底层系统调用崩掉。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

537

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int的含义
C++中int的含义

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

194

2025.08.29

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

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

67

2025.10.23

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

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

67

2025.10.23

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

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

480

2023.08.10

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

0

2026.01.14

php与html混编教程大全
php与html混编教程大全

本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

12

2026.01.13

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
极客学院Java8新特性视频教程
极客学院Java8新特性视频教程

共17课时 | 3.7万人学习

Git 教程
Git 教程

共21课时 | 2.7万人学习

Excel 教程
Excel 教程

共162课时 | 11.7万人学习

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

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