0

0

C# 无锁编程技术 C#如何使用Interlocked实现无锁算法

畫卷琴夢

畫卷琴夢

发布时间:2026-02-15 08:38:04

|

280人浏览过

|

来源于php中文网

原创

interlocked能替代lock因其直接编译为cpu原子指令,无系统调度开销,适用于int/long等类型的单次读改写;但不支持浮点运算、复合逻辑或对象状态保护,仅适合计数器、cas场景。

c# 无锁编程技术 c#如何使用interlocked实现无锁算法

Interlocked 为什么能替代 lock

因为 Interlocked 系列方法(如 IncrementAddCompareExchange)在底层直接编译为 CPU 的原子指令(如 x86 的 XADDCMPXCHG),不需要操作系统介入线程调度或获取互斥锁,天然避免了上下文切换和锁竞争开销。它只适用于简单类型(intlongrefbool 等)的单次读-改-写操作,不能用于复合逻辑(比如“先读再判断再写”这种多步操作,除非用 CompareExchange 手动实现 CAS 循环)。

什么时候该用 Interlocked.Increment 而不是 lock

典型场景是计数器累加——比如统计请求量、缓存命中次数、对象创建总数等纯数值递增需求。此时 Interlocked.Increment(ref count)lock(obj) { count++; } 更轻量、无阻塞、无死锁风险。

  • count 必须是 intlong 类型的字段(不能是属性,也不能是局部变量,必须可寻址)
  • 不要试图用它保护多个变量:比如同时更新 sumcount,这无法原子保证一致性
  • 返回值有意义:Interlocked.Increment 返回的是**递增后的值**,可直接用于条件判断(如限流:if (Interlocked.Increment(ref reqCount) > 100) Reject())

用 Interlocked.CompareExchange 实现无锁栈或计数器条件更新

这是唯一能模拟“CAS(Compare-And-Swap)”行为的方法,适用于需要“读-判断-写”原子语义的场景。比如实现一个线程安全的懒初始化标志:

MusicArt
MusicArt

AI音乐生成器

下载
private int _initialized = 0; // 0=未初始化,1=已初始化
public void EnsureInitialized()
{
    if (Interlocked.CompareExchange(ref _initialized, 1, 0) == 0)
    {
        // 当前值是 0,成功设为 1,说明本线程首次执行初始化
        InitializeCore();
    }
}

关键点:

  • CompareExchange(ref location, newValue, comparand) 只有当 location == comparand 时才把 newValue 写入,并返回原始值;否则不写,只返回当前值
  • 必须用循环重试才能实现真正可靠的无锁结构(如无锁队列),单次调用不等于“一定成功”,失败后需重新读取最新值再试
  • 不能传入计算表达式作为参数(如 Interlocked.CompareExchange(ref x, x + 1, x) 是错的——x 在两次求值间可能已被其他线程修改)

Interlocked 不支持哪些操作?常见误用陷阱

它不提供原子的乘法、除法、浮点运算,也不支持结构体或对象引用的深层比较(CompareExchange<t></t> 只比较引用地址,不是内容)。以下写法都是危险或无效的:

  • doubledecimal 调用 Interlocked.Add —— 编译报错,.NET 不提供浮点原子操作
  • Interlocked.Exchange(ref obj, new MyObj()) 替代锁来保护对象状态 —— 这只原子替换引用,但 MyObj 内部字段仍可能被多线程并发修改
  • foreach 遍历集合时用 Interlocked 修改集合长度字段 —— 集合本身不是线程安全的,仅改长度毫无意义,且可能破坏内部结构
  • Interlocked.Read(ref longValue) 用在非 long 字段上 —— 它只接受 ref long,且仅在 32 位系统上对 long 读取有必要(64 位系统上普通读已是原子)

真正的无锁编程难点不在调用几个 Interlocked 方法,而在于设计出能用有限原子原语表达的正确并发逻辑——多数业务场景下,老实用 lock 反而更安全、更易维护。

相关文章

编程速学教程(入门课程)
编程速学教程(入门课程)

编程怎么学习?编程怎么入门?编程在哪学?编程怎么学才快?不用担心,这里为大家提供了编程速学教程(入门课程),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

810

2023.08.22

counta和count的区别
counta和count的区别

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

199

2023.11.20

php中foreach用法
php中foreach用法

本专题整合了php中foreach用法的相关介绍,阅读专题下面的文章了解更多详细教程。

162

2025.12.04

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

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

342

2025.06.09

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

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

198

2025.07.04

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

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

342

2025.06.09

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

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

198

2025.07.04

string转int
string转int

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

730

2023.08.02

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

76

2026.02.13

热门下载

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

精品课程

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

共94课时 | 9.5万人学习

C 教程
C 教程

共75课时 | 4.7万人学习

C++教程
C++教程

共115课时 | 17.8万人学习

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

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