0

0

C# Interlocked类使用方法 C#如何实现原子操作

幻夢星雲

幻夢星雲

发布时间:2026-03-14 16:27:31

|

743人浏览过

|

来源于php中文网

原创

Interlocked不能直接操作普通变量,因其所有方法均要求ref参数以获取内存地址来执行CPU原子指令;Increment/Decrement仅支持±1整数增减,Add支持任意增量且返回新值;CompareExchange是CAS基础,用于实现无锁编程;Read用于long原子读,Or/Xor用于标志位操作。

c# interlocked类使用方法 c#如何实现原子操作

Interlocked 为什么不能直接操作普通变量

因为 Interlocked 的所有方法都要求传入 ref intref long 等可寻址的变量,它底层依赖 CPU 的原子指令(如 XCHGLOCK XADD),必须能拿到变量的内存地址。如果你传一个属性、临时计算值或只读字段,编译器会报错:Cannot pass 'xxx' as a ref or out argument because it is not a variable

实操建议:

  • 确保操作的是类字段或局部变量,且类型为 intlongIntPtrobject 或泛型 T(.NET 6+ 的 Interlocked.CompareExchange<t></t>
  • 不要试图对 struct 字段直接调用,比如 Interlocked.Increment(ref myObj.Counter) 是合法的,但 Interlocked.Increment(ref myObj.SomeStruct.Value) 可能因不可寻址而失败
  • 避免在属性 getter 中包装 Interlocked 调用——这掩盖了线程安全边界,也容易误以为“属性本身是线程安全的”

Increment/Decrement 和 Add 的区别在哪

Interlocked.IncrementInterlocked.Decrement 是特化版本,语义清晰、性能略优,仅支持 ±1;而 Interlocked.Add 更通用,支持任意整数增量,且在 .NET Core 2.0+ 后已支持 longint,还新增了 Interlocked.Add(ref long, long) 的重载。

常见错误现象:用 Increment 实现计费扣减 0.5 元?不行,它只接受整数。此时必须用 Interlocked.Add(ref balance, -50)(单位为分)或改用 double 配合 CompareExchange 手动实现。

实操建议:

  • 计数器类场景(如请求量统计)优先用 Increment/Decrement,语义直白,JIT 也可能做额外优化
  • 金额、偏移量等需非 ±1 修改时,统一用 Interlocked.Add,别自己写循环 + CompareExchange
  • 注意 Add 返回的是**变更后**的值,不是旧值(这点和 Increment 一致)

CompareExchange 是万能原子操作基座

几乎所有高级原子逻辑(如无锁栈、CAS 循环、延迟初始化)都基于 Interlocked.CompareExchange。它本质是:“如果当前值等于预期值,就设成新值,并返回原值;否则只返回当前值,不修改”。这是唯一能实现“读-改-写”原子性的入口。

使用场景举例:实现线程安全的懒加载单例

PixVerse
PixVerse

PixVerse是一款强大的AI视频生成工具,可以轻松地将多种输入转化为令人惊叹的视频。

下载
private static MyService _instance;
private static readonly object _lock = new();
public static MyService Instance
{
    get
    {
        if (_instance == null)
        {
            var instance = new MyService();
            Interlocked.CompareExchange(ref _instance, instance, null);
        }
        return _instance;
    }
}

这个写法比 lock 轻量,但要注意:多个线程可能同时创建 MyService 实例,只有第一个成功写入的被保留,其余被丢弃(所以构造函数不能有副作用)。

实操建议:

  • 写 CAS 循环时,务必把 CompareExchange 放在 do-while 内部,反复读取当前值并尝试更新
  • .NET 6+ 提供了 Interlocked.CompareExchange<t>(ref T, T, T)</t>,可用于引用类型或不可变结构体,但要求 T 是无指针类型(no unsafe pointers)
  • 不要用 CompareExchange 操作浮点数(float/double)做精确比较——NaN、-0.0 等会导致意外交替失败

Read、Or、Xor 这些冷门方法什么时候用

Interlocked.Read 专用于 long 在 32 位环境下的原子读取(x86 下 long 读写非原子,可能读到撕裂值),.NET Core / .NET 5+ 在 x64 上已默认原子,但仍建议保留以保兼容;OrXor 常用于标志位管理,比如多线程协同设置状态掩码。

性能影响:这些操作都是单条 CPU 指令,开销远低于锁,但频繁调用仍会引发缓存行争用(false sharing)。例如把几个高频更新的 int 字段放在同一个类里,它们可能共享 L1 缓存行,导致性能反降。

实操建议:

  • 跨平台项目中读 long 字段,坚持用 Interlocked.Read(ref field),别依赖运行时架构判断
  • Or 设置标志位(如 Interlocked.Or(ref flags, (int)Flags.Ready)),用 Xor 切换开关(如翻转调试模式)
  • 若字段间无逻辑关联,给它们加 [StructLayout(LayoutKind.Sequential)][FieldOffset] 手动隔离,或插入 long 填充字段防 false sharing
真正难的不是调用哪个方法,而是判断「这里是否真的需要原子操作」——很多场景其实只需要 volatile 读写,或者根本该用 ConcurrentQueue 这类高层抽象。过早手写 Interlocked 循环,反而容易漏掉内存屏障语义或引入 ABA 问题。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
css中float用法
css中float用法

css中float属性允许元素脱离文档流并沿其父元素边缘排列,用于创建并排列、对齐文本图像、浮动菜单边栏和重叠元素。想了解更多float的相关内容,可以阅读本专题下面的文章。

595

2024.04.28

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

108

2025.10.23

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

107

2023.09.25

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

490

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

202

2025.07.04

string转int
string转int

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

1051

2023.08.02

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

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

615

2024.08.29

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

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

335

2025.08.29

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 11.3万人学习

C 教程
C 教程

共75课时 | 5.4万人学习

C++教程
C++教程

共115课时 | 21.8万人学习

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

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