0

0

C#怎么实现多线程下的安全累加_C#如何使用原子操作类【代码】

尼克

尼克

发布时间:2026-03-18 13:33:33

|

971人浏览过

|

来源于php中文网

原创

Interlocked.Add是最直接的安全累加方案,因其将“读-改-写”封装为CPU级原子指令,无需锁且返回操作后新值,适用于计数器等简单整数累加场景。

为什么 Interlocked.Add 是最直接的安全累加方案

因为普通 +++= 在多线程下会丢失更新——两个线程同时读取同一变量值、各自加1、再写回,结果只加了1次而不是2次。interlocked.add 把“读-改-写”变成一条 cpu 级原子指令,不依赖锁,也不阻塞线程。

适用场景:计数器、统计总耗时、累计错误次数等无需复杂同步逻辑的整数累加。

  • Interlocked.Add 只支持 intlonguintulong 和指针类型;不能用于 double 或自定义类型
  • 它返回的是**操作后的最新值**,不是旧值(这点和 Interlocked.Increment 一致)
  • 在 .NET Core 3.0+ 和 .NET 5+ 中,对 long 的操作在 x64 上是原生原子,在 x86 上由 CLR 保证原子性,不用额外处理
int counter = 0;
// 多个线程并发调用
Interlocked.Add(ref counter, 1); // 安全
Interlocked.Add(ref counter, 5); // 同样安全

别用 lock 做简单累加,除非你真需要复合操作

lock 实现累加虽然能保证安全,但引入了线程等待、上下文切换和锁竞争开销。对单个整数累加这种极轻量操作,性能差距明显——实测在高并发下,Interlocked.Addlock 快 3–5 倍以上。

常见误用:为“以后可能要扩展逻辑”提前上锁,结果把简单路径拖慢。

  • 如果只是 counter += value,坚决用 Interlocked.Add
  • 如果要同时更新多个变量(比如 sumcount),或者需要判断后再更新(如“仅当小于阈值才加”),才考虑 lockSpinLock
  • lock 对象别用 this 或字符串字面量,容易引发意外死锁或跨实例干扰

Interlocked.CompareExchange 能做更灵活的条件累加

当累加需要前置检查(比如“只在当前值小于 100 时才加 1”),Interlocked.Add 不够用,这时得靠 CompareExchange 手动实现 CAS 循环。

它本质是:比较当前值是否等于预期值,是则替换为新值并返回旧值;否则返回当前实际值,由你决定是否重试。

Buildt.ai
Buildt.ai

AI驱动的软件开发平台,可以自动生成代码片段、代码分析及其他自动化任务

下载
  • 必须用循环包裹,因为可能被其他线程抢先修改导致失败
  • 避免在循环里做耗时操作,否则浪费 CPU
  • 注意不要无脑死循环,可加简单重试次数限制或 Thread.Yield()
int threshold = 100;
int current, newValue;
do {
    current = counter;
    if (current >= threshold) break;
    newValue = current + 1;
} while (Interlocked.CompareExchange(ref counter, newValue, current) != current);

别忽略 volatile 和字段声明的影响

Interlocked 操作本身已包含内存屏障语义,所以累加目标字段**不需要**加 volatile 修饰符——加了反而误导,让人以为靠它就能保证线程安全。

真正要注意的是字段声明位置和生命周期:

  • 静态字段累加需确认是否跨 AppDomain(.NET Framework)或 AssemblyLoadContext(.NET Core+),否则可能不是同一个变量
  • 实例字段累加时,确保所有线程操作的是同一个对象实例,而不是各自 new 出来的副本
  • 局部变量不能用 ref 传给 Interlocked 方法——编译器会报错,这是硬性限制

原子操作不是银弹。它快、轻量、适合整数,但也仅限于此。一旦涉及浮点运算、结构体更新或业务规则耦合,就得换思路——这时候想“怎么加得更安全”,不如先问“这个累加逻辑真的该放在这里吗”。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

203

2023.11.20

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

761

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1571

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

652

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1269

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

1226

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

194

2025.07.29

Python WebSocket实时通信与异步服务开发实践
Python WebSocket实时通信与异步服务开发实践

本专题聚焦 Python 在实时通信场景中的开发实践,系统讲解 WebSocket 协议原理、长连接管理、消息推送机制以及异步服务架构设计。内容包括客户端与服务端通信实现、连接稳定性优化、消息队列集成及高并发处理策略。通过完整案例,帮助开发者构建高效稳定的实时通信系统,适用于聊天应用、实时数据推送等场景。

3

2026.03.18

热门下载

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

精品课程

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

共94课时 | 11.6万人学习

C 教程
C 教程

共75课时 | 5.6万人学习

C++教程
C++教程

共115课时 | 22.4万人学习

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

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