Monitor.IsEntered 是检查当前线程是否持有指定对象的 Monitor 锁,仅用于调试和异常兜底清理,不能用于同步逻辑判断,因其不保证原子性、无跨线程可见性且不参与实际锁操作。
Monitor.IsEntered 是一个静态方法,用于检查当前线程是否已对指定对象获取了 Monitor 锁(即是否已执行过 Monitor.Enter 且尚未配对调用 Monitor.Exit)。但它**不能安全用于同步逻辑判断**——它只反映“当前线程是否持有该对象的 Monitor 锁”,不反映其他线程是否持有、是否正在等待、或锁是否已被释放。
常见误用是想靠它实现“如果没锁就加锁”,类似:
if (!Monitor.IsEntered(obj)) { Monitor.Enter(obj); } 这种写法存在竞态:IsEntered 返回 false 后,另一线程可能立刻 Enter,导致本线程仍会阻塞在后续 Enter 上,且无法保证原子性。
它的实用价值集中在开发期诊断和极少数需要“自救”的异常处理中。例如在 finally 块里做防御性 Exit,但又不确定是否真的 Enter 过:
Exit 抛出 SynchronizationLockException
catch 中尝试清理锁时防止崩溃示例(安全的 finally 清理):
object lockObj = new object();
try
{
Monitor.Enter(lockObj);
// 可能抛异常的临界区操作
}
finally
{
if (Monitor.IsEntered(lockObj))
Monitor.Exit(lockObj);
}
lock(obj) { ... } 编译后自动展开为带 try/finally 的 Monitor.Enter/Exit,确保即使异常也能释放锁。Monitor.IsEntered 本身不参与任何同步语义,它既不加锁也不放锁,也不影响其他线程行为。
使用它来“绕开”标准锁模式,往往意味着逻辑已变得难以追踪。尤其要注意:
true 仅当本线程对同一对象调用了 Enter 且未 Exit;嵌套 Enter 也会返回 true,但 Exit 必须配对次数才能完全释放async 方法无效——await 后续可能在不同线程执行,IsEntered 在新线程上永远返回 false
Monitor 对不可重入锁做了更多优化,但 IsEntered 行为未变,依然不提供跨线程可见性保证绝大多数需要“条件加锁”或“尝试加锁”的场景,应直接使用更明确的原语:
Monitor.TryEnter(obj, 0) 或 Monitor.TryEnter(obj, timeout)
SemaphoreSlim.WaitAsync()(注意它不是 Monitor 的替代,但支持 async)ThreadLocal + 计数管理,或评估是否真需可重入(多数情况不需要)Monitor.IsEntered 留给调试输出或极端兜底清理即可,把它当成生产代码里的控制分支
