0

0

c# 线程安全和非线程安全的集合有什么区别

煙雲

煙雲

发布时间:2025-12-29 10:59:39

|

363人浏览过

|

来源于php中文网

原创

线程安全集合与非线程安全集合的本质区别在于多线程同时读写时是否保证数据一致性和不崩溃;ConcurrentDictionary等通过分段锁、原子操作和线程本地存储实现高效并发,而Dictionary等在并发写入时会抛出异常或导致数据错乱。

c# 线程安全和非线程安全的集合有什么区别

线程安全集合 vs 非线程安全集合:本质区别在哪

核心区别不是“能不能用”,而是“多线程同时读写时会不会崩”。ConcurrentDictionary 允许多个线程同时 AddTryGetValueRemove,不加锁也不抛 InvalidOperationException;而 Dictionary 在同样场景下大概率崩溃——不是每次必现,但只要发生,就是数据错乱或 Collection was modified 异常。

System.Collections.Concurrent 里的集合怎么做到“不加锁也安全”

它们不用粗粒度的全局锁(比如整个 lock(_dict)),而是靠细粒度控制 + 无锁原子操作:

  • ConcurrentQueueConcurrentStack 完全不用锁,靠 Interlocked.CompareExchange 等 CPU 原子指令完成入队/出
  • ConcurrentDictionary 把内部哈希桶分段(默认 31 段),写不同段互不影响;同一段内才用轻量级 SpinLock,避免线程挂起开销
  • ConcurrentBag 为每个线程维护本地队列(ThreadLocalList),添加/取自己线程的数据几乎零竞争;跨线程“偷取”时才做同步

所以它不是“慢但稳”,而是“快且稳”——尤其在高并发、读写混合场景下,比手动用 lock 包裹 List 性能高出数倍。

别误用 Synchronized 包装器:它不是线程安全的“快捷方式”

ArrayList.SynchronizedHashtable.Synchronized 这类老式包装器,只是给每个方法加了同一个对象锁。问题很实在:

  • 所有操作(AddRemoveCount、遍历)都抢同一把锁 → 严重串行化,吞吐量暴跌
  • 看似“线程安全”,但像 if (list.Count > 0) item = list[0]; 这种两步操作,中间可能被其他线程修改 → 仍是竞态条件(race condition)
  • 它属于 System.Collections 非泛型体系,还有装箱/拆箱开销和类型不安全问题

结论:新项目里完全不要用 Synchronized 包装器,直接上 System.Collections.Concurrent 的泛型并发集合。

千博购物系统.Net
千博购物系统.Net

千博购物系统.Net能够适合不同类型商品,为您提供了一个完整的在线开店解决方案。千博购物系统.Net除了拥有一般网上商店系统所具有的所有功能,还拥有着其它网店系统没有的许多超强功能。千博购物系统.Net适合中小企业和个人快速构建个性化的网上商店。强劲、安全、稳定、易用、免费是它的主要特性。系统由C#及Access/MS SQL开发,是B/S(浏览器/服务器)结构Asp.Net程序。多种独创的技术使

下载

什么时候该用非线程安全集合

不是“不能用”,而是“不该在共享上下文中用”。适用场景非常明确:

  • 单线程逻辑:比如 MVC Controller 里构造一个 List 做临时计算,返回前就丢弃
  • 只读共享:多个线程只调用 ReadOnlyCollection.AsReadOnly(list)IEnumerable 遍历,且确保源集合创建后不再修改
  • 局部变量 or 方法内集合:生命周期严格绑定当前线程栈,无跨线程传递
  • 性能敏感且已用 lock / ReaderWriterLockSlim 手动保护的临界区(此时用 Dictionary 可能比 ConcurrentDictionary 更省内存)

最容易踩的坑是:以为“我只读不写就安全”,结果忘了某个地方悄悄调了 ToList()ToArray(),返回的是新集合——但原始集合若还在被其他线程写,那这个“只读快照”本身就不一致。

真正要记住的不是“哪个类安全”,而是“谁在访问、怎么访问、生命周期归谁管”。ConcurrentBag 不是万能解药,List 也不是洪水猛兽——关键在上下文。

相关专题

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

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

709

2023.08.22

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

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

192

2023.11.20

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

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

364

2023.07.18

堆和栈区别
堆和栈区别

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

558

2023.08.10

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

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

465

2023.08.10

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

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

63

2025.12.24

Golang 命令行工具(CLI)开发实战
Golang 命令行工具(CLI)开发实战

本专题系统讲解 Golang 在命令行工具(CLI)开发中的实战应用,内容涵盖参数解析、子命令设计、配置文件读取、日志输出、错误处理、跨平台编译以及常用CLI库(如 Cobra、Viper)的使用方法。通过完整案例,帮助学习者掌握 使用 Go 构建专业级命令行工具与开发辅助程序的能力。

1

2025.12.29

ip地址修改教程大全
ip地址修改教程大全

本专题整合了ip地址修改教程大全,阅读下面的文章自行寻找合适的解决教程。

162

2025.12.26

压缩文件加密教程汇总
压缩文件加密教程汇总

本专题整合了压缩文件加密教程,阅读专题下面的文章了解更多详细教程。

52

2025.12.26

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 0.9万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

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

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