0

0

c# ConcurrentBag 和 List 加 lock 的区别

星降

星降

发布时间:2026-01-18 10:34:02

|

747人浏览过

|

来源于php中文网

原创

ConcurrentBag 通过线程本地包+全局包实现无锁增删,高并发写入性能优,但遍历需全量拷贝;lock(List)语义清晰但锁粒度大;选型应依读写比例、顺序需求及访问模式而定。

c# concurrentbag 和 list 加 lock 的区别

ConcurrentBag 不用 lock 就线程安全,但遍历很贵

它内部用的是每个线程私有本地包(ThreadLocal bag)+ 全局共享包的混合结构,添加(Add)和尝试取(TryTake)基本不争抢,所以高并发写入时比 lock 包裹的 List 快很多。但注意:foreach 遍历 ConcurrentBag 会先拷贝全部元素到一个临时 List,再枚举——这意味着每次遍历都触发一次内存分配 + O(n) 拷贝。如果你的场景是“多线程狂塞、单线程最后扫一遍”,那遍历前手动转成 List 更划算:

var snapshot = new List(bag); // 一次性拷贝
foreach (var p in snapshot)
{
    Console.WriteLine(p.Name);
}

lock(List) 简单可控,但锁粒度大、易成瓶颈

object 锁住整个 List 实例,所有读写(AddRemoveAtCount、甚至 foreach)都排队执行。好处是语义清晰、调试方便;坏处是:哪怕只是读 Count,也要等前面的写操作释放锁;多个线程同时调用 Add 会严重串行化。

  • 别在 lock 块里做耗时操作(比如 IO、网络请求),否则锁持有时间拉长,拖垮整体吞吐
  • 不要用 list 本身当锁对象(lock(list)),它可能被外部修改或设为 null,推荐用专用 private readonly object _lock = new object();
  • ListForEach 方法不是线程安全的——即使加了 lock,遍历时若其他线程正修改,仍可能抛 InvalidOperationException

选哪个?看你的读写比例和访问模式

不是“ConcurrentBag 一定比 lock(List) 快”,而是看实际行为:

Autoppt
Autoppt

Autoppt:打造高效与精美PPT的AI工具

下载
  • 写远多于读(如日志缓冲、事件暂存)→ ConcurrentBag 明显优势
  • 读多写少(如配置缓存、只偶尔更新的元数据)→ ReaderWriterLockSlim + List 可能更优
  • 需要按索引随机访问(list[i])、频繁中间插入/删除 → ConcurrentBag 不支持,只能换思路(比如改用 ConcurrentDictionary 模拟索引)或坚持 lock
  • 必须保持插入顺序且 FIFO 处理 → 别用 ConcurrentBag(它是无序的),改用 ConcurrentQueue

容易被忽略的坑:ConcurrentBag 的“无序”不是 bug,是设计

ConcurrentBag 不保证任何顺序:AddTryTake 的结果取决于线程本地包状态和全局包竞争,同一个线程连续 Add 两个元素,TryTake 也可能先拿到后一个。如果你依赖顺序(比如任务队列、流水线阶段),用它就埋了隐性 bug。这时候宁可多花点性能成本,也该选 ConcurrentQueue 或带锁的 List + 手动维护索引。

另外,ConcurrentBagToArray()ToList() 同样要全量拷贝,别在热路径反复调用。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

231

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

436

2024.03.01

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

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

197

2023.11.20

php中foreach用法
php中foreach用法

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

42

2025.12.04

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

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

481

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

143

2025.12.24

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

72

2026.01.16

全民K歌得高分教程大全
全民K歌得高分教程大全

本专题整合了全民K歌得高分技巧汇总,阅读专题下面的文章了解更多详细内容。

131

2026.01.16

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

54

2026.01.16

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

光速学会docker容器
光速学会docker容器

共33课时 | 1.9万人学习

时间管理,自律给我自由
时间管理,自律给我自由

共5课时 | 0.8万人学习

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

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