0

0

c# 如何用C#实现一个无锁(Lock-Free)的栈或队列

畫卷琴夢

畫卷琴夢

发布时间:2026-01-06 08:29:02

|

805人浏览过

|

来源于php中文网

原创

C# 的 ConcurrentStack 和 ConcurrentQueue 不是纯无锁,因高争用时会退化为 SpinWait 或 Monitor.TryEnter;其内部受限于 .NET 内存模型、GC 干预及缺乏双字原子操作,难以安全实现托管引用类型的 lock-free 队列。

c# 如何用c#实现一个无锁(lock-free)的栈或队列

为什么 C# 的 ConcurrentStackConcurrentQueue 不是纯无锁

它们对外宣称“线程安全”,但内部在高争用或内存压力大时会退化为使用 SpinWait 甚至轻量级锁(如 Monitor.TryEnter),并非严格意义上的 lock-free。真正的 lock-free 要求:任意线程被挂起、终止,都不影响其他线程继续完成操作(即满足 wait-free 或至少 lock-free 的 progress guarantee)。C# 运行时和 .NET 内存模型对无锁编程的支持有限,尤其涉及 GC 和引用类型时,CompareExchange 对对象引用的 ABA 问题无法靠硬件指令直接解决。

Interlocked.CompareExchange 实现 lock-free (仅限值类型)

这是最可控的起点。栈的 push/pop 只需原子更新头指针,适合用单个字段 + CAS 模拟。但必须避免 ABA —— 对值类型(如 intlong)可借助“带版本号的指针”(如 long 高 32 位存版本,低 32 位存地址),但 .NET 不支持原生双字 CAS(Interlocked.CompareExchange128 仅 Windows x64 且不支持托管引用)。所以实用方案是:只对值类型实现,或接受一定 ABA 风险(在低并发、短生命周期场景下可接受)。

  • 定义节点结构体:struct Node { public T Value; public Node* Next; }(需 unsafe
  • 头指针用 Node* 类型的字段,初始为 null
  • push:读当前 head → 构造新节点 → Interlocked.CompareExchange(ref head, newNode, oldHead) 循环直到成功
  • pop:同理,读 head → 若非空则用 CompareExchange 尝试更新为 head->Next
unsafe
{
    Node* head = null;
    Node* newNode = stackalloc Node[1];
    newNode->Value = value;
Node* oldHead;
do
{
    oldHead = head;
    newNode-youjiankuohaophpcnNext = oldHead;
} while (Interlocked.CompareExchange(ref head, newNode, oldHead) != oldHead);

}

托管引用类型下的无锁队列为何极难安全实现?

队列需要两个指针(head/tail),必须保证二者协同更新的原子性。CAS 单指针无法避免“tail 已前进但 head 未更新”的中间态被其他线程误读。Michael-Scott 队列算法虽经典,但在 .NET 中面临三个硬伤:

创一AI
创一AI

AI帮你写短视频脚本

下载
  • GC 可能在 CAS 成功后、新节点被其他线程读取前回收该节点(需 GCHandle.Alloc 固定,但开销大且易泄漏)
  • 引用类型的 ABA:节点 A 被弹出 → 内存复用 → 新节点 A' 分配在同一地址 → CAS 误判为“仍是原 A”
  • .NET 没有 atomic> 等双字原子操作,无法规避 tail/head 不一致读

因此,除非你控制整个生命周期(如对象池 + 手动内存管理 + unsafe + fixed 数组),否则不要尝试自己写托管无锁队列。

实际项目中该选什么?

绝大多数场景下,ConcurrentStackConcurrentQueue 是正确选择。它们在多数负载下表现接近无锁,且经过充分测试。只有当你明确观测到:Monitor.Enter 在性能剖析中成为瓶颈、且能接受 unsafe + 手动内存管理 + 仅限值类型 + 无 GC 干预时,才考虑手写。另外,.NET 6+ 的 Channel 在生产者/消费者模式下比手写无锁结构更可靠、更易维护。

真正容易被忽略的是:无锁 ≠ 更快。它只是消除了阻塞,但增加了缓存一致性流量(false sharing)、重试开销和实现复杂度。先用 dotnet-trace 确认锁确实是瓶颈,再决定是否踏入这个领域。

相关专题

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

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

194

2025.06.09

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

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

186

2025.07.04

string转int
string转int

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

314

2023.08.02

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

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

528

2024.08.29

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

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

49

2025.08.29

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

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

193

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

380

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

566

2023.08.10

PPT动态图表制作教程大全
PPT动态图表制作教程大全

本专题整合了PPT动态图表制作相关教程,阅读专题下面的文章了解更多详细内容。

12

2026.01.07

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
HTML5/CSS3/JavaScript/ES6入门课程
HTML5/CSS3/JavaScript/ES6入门课程

共102课时 | 6.6万人学习

前端基础到实战(HTML5+CSS3+ES6+NPM)
前端基础到实战(HTML5+CSS3+ES6+NPM)

共162课时 | 18.6万人学习

第二十二期_前端开发
第二十二期_前端开发

共119课时 | 12.2万人学习

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

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