0

0

c# Monitor.TryEnter 的超时功能和自旋等待

煙雲

煙雲

发布时间:2026-01-25 08:05:34

|

844人浏览过

|

来源于php中文网

原创

Monitor.TryEnter 需超时参数以防无限阻塞:超时为0即瞬时尝试,负数等价于无超时(不推荐),单位毫秒;其内部自旋由CLR自动控制且不可干预,需手动组合SpinWait与TryEnter(0)实现可控重试。

c# monitor.tryenter 的超时功能和自旋等待

Monitor.TryEnter 为什么需要超时参数

不加超时的 Monitor.Enter 会无限阻塞,一旦锁被长期占用(比如持有锁的线程崩溃、死锁或执行过久),调用方就彻底卡住,无法响应、无法释放资源、无法优雅降级。而 Monitor.TryEnter(object, int) 的超时能力,本质是给同步操作加了一道“安全阀”——它让线程最多等待指定毫秒数,超时后直接返回 false,而不是死等。

常见错误现象包括:服务接口偶发长时间 hang、后台任务线程池耗尽、健康检查失败但无明确异常日志——这些背后往往藏着未设超时的 Monitor.Enter

  • 超时值为 0 表示“只尝试一次,不等待”,类似自旋检测,成功则返回 true,否则立刻返回 false
  • 超时值为负数(如 -1)等价于无超时的 Monitor.Enter,**不推荐使用**
  • 超时单位是毫秒,不是 ticks 或秒;传入 1000 就是 1 秒,不是 1000ms 的近似值

Monitor.TryEnter 的自旋行为到底由谁控制

Monitor.TryEnter 本身不暴露自旋开关,它的自旋逻辑是 .NET 运行时内部实现的,且仅在特定条件下触发:当锁处于“轻量级”状态(即无竞争或刚释放)、且等待时间极短(通常几微秒内)时,CLR 可能先做几次 CPU 自旋(spin-wait),再转入真正的内核等待。这个过程对开发者透明,也无法通过参数干预。

这意味着你不能靠 TryEnter 实现可控的自旋重试策略。如果业务需要“最多自旋 100 次,每次 10 微秒,失败再退避”,必须手动写循环 + Thread.SpinWait + TryEnter(0) 组合:

bool acquired = false;
for (int i = 0; i < 100 && !acquired; i++)
{
    acquired = Monitor.TryEnter(lockObj, 0);
    if (!acquired)
        Thread.SpinWait(10);
}
if (!acquired)
{
    // 转入带超时的等待,或放弃
    acquired = Monitor.TryEnter(lockObj, 50);
}

注意:Thread.SpinWait(n) 中的 n 是提示值,实际时长由 CPU 频率和调度决定,不可精确控制。

迦恩计算机资源网源码(图书销售类)
迦恩计算机资源网源码(图书销售类)

采用三层架构开发,前台集成了产品在线展示,用户注册、在线调查、在线投稿后台有类别管理\图书管理\订单管理\会员管理\配送范围管理\邮件列表\广告管理\友情链接管理等后台添加图书时自动生成缩略图和文字水印主要参考了petshop的设计架构、使用了Asp.net2.0中很多MemberShip、master等新功能后台管理地址/web/admin/ 超级管理员账号密码均为aspx1特别提示:该系统需要

下载

超时值设多大才合理

没有通用答案,取决于临界区执行时间和系统 SLA。设得太小会导致频繁抢锁失败,设得太大又失去超时意义。关键判断依据是:「这个锁保护的操作,在正常情况下应该多久完成?」

  • 纯内存操作(如修改几个字段)→ 通常 1–10 ms 足够,超时可设 50 ms
  • 涉及简单 IO 缓存读取(如 ConcurrentDictionary 查找)→ 建议 100 ms 起步
  • 任何可能触发 GC、远程调用、磁盘访问的临界区 → 不该放在 Monitor 里,应重构;若必须,超时至少设为该操作 P95 延迟的 2–3 倍
  • 永远不要设成 Timeout.Infinite(即 -1)——它等于放弃超时保障

TryEnter 返回 false 后的典型误操作

很多开发者把 TryEnter 当成“尽力而为”,返回 false 就直接跳过逻辑,导致数据不一致或功能缺失。更危险的是在 false 后仍继续访问被保护资源:

if (Monitor.TryEnter(_lockObj, 100))
{
    try
    {
        _sharedCounter++; // 安全
    }
    finally
    {
        Monitor.Exit(_lockObj);
    }
}
else
{
    _sharedCounter++; // ❌ 危险!未持锁就修改
}

正确做法只有三种:

  • 明确允许“跳过”(如日志缓冲刷写失败可丢弃)→ 确保跳过逻辑本身无副作用
  • 降级到其他同步机制(如用 SpinLock无锁结构)
  • 抛出异常或返回错误码,让上层决定重试/熔断/告警

最容易被忽略的一点:超时不是性能问题的遮羞布。如果 TryEnter(..., 100) 频繁返回 false,说明锁争用已成瓶颈,该优化临界区代码、拆分锁粒度,而不是不断调高超时值。

相关专题

更多
string转int
string转int

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

401

2023.08.02

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

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

543

2024.08.29

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

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

53

2025.08.29

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

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

197

2025.08.29

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1072

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

148

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1010

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

13

2026.01.19

c++ 根号
c++ 根号

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

58

2026.01.23

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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