c# 线程同步的几种方式

发布时间 - 2026-01-11 00:00:00    点击率:
lock最常用但易错,需注意锁对象生命周期、避免死锁和粒度控制;Monitor.TryEnter支持超时;ReaderWriterLockSlim适合读多写少;Interlocked适用于简单原子操作。

lock 语句是最常用也最容易出错的同步方式

绝大多数 C# 开发者第一反应就是 lock,它底层基于 Monitor.Enter / Monitor.Exit,用起来简单但有几个关键点必须注意:

  • lock 的对象必须是引用类型,且生命周期要稳定——不能是 new object() 放在方法内(每次新建不同实例,锁失效),更不能是值类型(会装箱成不同对象)
  • 避免锁住 thistypeof(XXX) 或公共静态对象,容易引发外部死锁或意外争用
  • 锁内不要调用可能阻塞或重入的代码(比如再调用另一个也用 lock 的方法),否则极易死锁
  • 锁的粒度要尽量小:只包裹真正需要保护的共享资源读写段,而不是整个方法体
private readonly object _syncLock = new object();
private int _counter = 0;

public void Increment() { lock (_syncLock) // ✅ 正确:私有、只读、引用类型字段 { _counter++; // 只包临界区 } }

Monitor.TryEnter 可控超时,适合避免无限等待

当无法确定锁何时能释放(比如依赖外部服务响应),硬等 lock 会导致线程挂起甚至拖垮系统。Monitor.TryEnter 提供带超时和可取消的入口控制:

  • 返回 bool 表示是否成功获取锁,不阻塞线程
  • 支持毫秒整数超时,也支持 TimeSpanCancellationToken
  • 必须配对调用 Monitor.Exit(即使没拿到锁也要确保不漏释放)
  • lock 多一层手动管理,但更灵活
private readonly object _syncLock = new object();

public bool TryProcessWithTimeout(int timeoutMs = 100) { if (Monitor.TryEnter(_syncLock, timeoutMs)) { try { // 执行临界区操作 return true; } finally { Monitor.Exit(_syncLock); // ✅ 必须保证释放 } } return false; // 超时未获取到锁 }

ReaderWriterLockSlim 适合读多写少的共享数据结构

如果某个对象被频繁读取、极少修改(比如配置缓存、路由表),用 lock 会让所有读操作排队,严重降低吞吐。ReaderWriterLockSlim 允许多个读线程并发,仅写时独占:

  • EnterReadLock / ExitReadLock:允许多个线程同时进入
  • EnterWriteLock / ExitWriteLock:写时阻塞所有读和写
  • 支持升级策略(EnterUpgradeableReadLock),在读锁内判断需写时再升级,减少锁竞争
  • 注意:它不支持递归获取(同一线程重复 Enter 会死锁),也不像 lock 那样自动释放
private readonly ReaderWriterLockSlim _rwLock = new ReaderWriterLockSlim();
private Dictionary _cache = new Dictionary();

public string Get(string key) { _rwLock.EnterReadLock(); try { return _cache.TryGetValue(key, out var value) ? value : null; } finally { _rwLock.ExitReadLock(); // ✅ 必须显式释放 } }

public void Set(string key, string value) { _rwLock.EnterWriteLock(); try { _cache[key] = value; } finally { _rwLock.ExitWriteLock(); } }

Interlocked 类适用于简单原子操作,零开销

当只需要对整数、引用或指针做「读-改-写」原子操作(如计数器、标志位、无锁栈头更新),Interlocked 是最优选——它直接编译为 CPU 原子指令,没有锁开销,也不涉及线程调度:

  • Interlocked.IncrementInterlocked.CompareExchangeInterlocked.Exchange 最常用
  • 只能用于 intlongIntPtr、引用类型等有限类型
  • 不能替代复杂逻辑同步(比如“如果余额 > 100 就扣款”,这需要 CompareExchange 循环重试,但业务逻辑一复杂就难维护)
  • 注意:它不提供内存屏障之外的同步语义,对非原子字段组合操作无效
private long _requestCount = 0;
private int _status = 0; // 0=stopped, 1=running

public void RecordRequest() => Interlocked.Increment(ref _requestCount);

public bool TryStart() { return Interlocked.CompareExchange(ref _status, 1, 0) == 0; // 仅当原值为 0 时设为 1 }

实际项目里,lock 覆盖 80% 场景;但一旦出现性能瓶颈或死锁,得立刻想到 Monitor.TryEnter 控制等待、ReaderWriterLockSlim 分离读写、或者用 Interlocked 拆解原子动作——这些不是“高级技巧”,而是线程安全落地时绕不开的权衡点。


#   # 路由  # c#  # 性能瓶颈  # 无锁  # 有锁  # Object  # 递归  # bool  # int  # 循环  # 指针  # 数据结构  # 值类型  # 引用类型  # 线程  # 并发  # 对象  # typeof  # this  # 死锁  # 最常用  # 多个  # 适用于  # 它不  # 多写  # 也不  # 放在  # 也要 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel的契約(Contracts)是什么_深入理解Laravel Contracts与依赖倒置  Laravel如何使用Scope本地作用域_Laravel模型常用查询逻辑封装技巧【手册】  如何快速查询网址的建站时间与历史轨迹?  Laravel如何实现全文搜索功能?(Scout和Algolia示例)  大连网站制作公司哪家好一点,大连买房网站哪个好?  JS碰撞运动实现方法详解  Laravel如何使用Contracts(契约)进行编程_Laravel契约接口与依赖反转  简单实现jsp分页  Laravel如何处理异常和错误?(Handler示例)  如何快速使用云服务器搭建个人网站?  关于BootStrap modal 在IOS9中不能弹出的解决方法(IOS 9 bootstrap modal ios 9 noticework)  java ZXing生成二维码及条码实例分享  Laravel如何操作JSON类型的数据库字段?(Eloquent示例)  laravel怎么为应用开启和关闭维护模式_laravel应用维护模式开启与关闭方法  Laravel怎么上传文件_Laravel图片上传及存储配置  浏览器如何快速切换搜索引擎_在地址栏使用不同搜索引擎【搜索】  如何快速搭建高效WAP手机网站吸引移动用户?  如何快速搭建安全的FTP站点?  Swift中循环语句中的转移语句 break 和 continue  Laravel如何实现模型的全局作用域?(Global Scope示例)  微信小程序 require机制详解及实例代码  图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?  Laravel如何构建RESTful API_Laravel标准化API接口开发指南  如何在VPS电脑上快速搭建网站?  Android中Textview和图片同行显示(文字超出用省略号,图片自动靠右边)  如何为不同团队 ID 动态生成多个“认领值班”按钮  Laravel策略(Policy)如何控制权限_Laravel Gates与Policies实现用户授权  python中快速进行多个字符替换的方法小结  如何在IIS7中新建站点?详细步骤解析  Laravel如何使用Socialite实现第三方登录?(微信/GitHub示例)  Laravel怎么为数据库表字段添加索引以优化查询  HTML透明颜色代码怎么让下拉菜单透明_下拉菜单透明背景指南【技巧】  javascript中的try catch异常捕获机制用法分析  Win11摄像头无法使用怎么办_Win11相机隐私权限开启教程【详解】  Laravel如何实现数据导出到PDF_Laravel使用snappy生成网页快照PDF【方案】  个人网站制作流程图片大全,个人网站如何注销?  php8.4header发送头信息失败怎么办_php8.4header函数问题解决【解答】  Laravel怎么在Controller之外的地方验证数据  Laravel定时任务怎么设置_Laravel Crontab调度器配置  佛山企业网站制作公司有哪些,沟通100网上服务官网?  移动端脚本框架Hammer.js  Bootstrap整体框架之JavaScript插件架构  Laravel API资源(Resource)怎么用_格式化Laravel API响应的最佳实践  Laravel如何实现用户密码重置功能?(完整流程代码)  Laravel如何使用Sanctum进行API认证?(SPA实战)  如何选择可靠的免备案建站服务器?  如何在宝塔面板创建新站点?  Linux网络带宽限制_tc配置实践解析【教程】  公司网站制作价格怎么算,公司办个官网需要多少钱?  如何在万网自助建站平台快速创建网站?