贝利信息

c# lock(new object()) 和 lock(static_object) 的区别

日期:2026-01-22 00:00 / 作者:煙雲
lock(new object())几乎没用,因为每次新建对象实例导致线程锁不同对象,无法实现互斥;正确做法是用private static readonly object _syncLock = new object()确保共享同一引用。

lock(new object()) 为什么几乎没用

它每次执行都新建一个 object 实例,而 lock 的作用是让多个线程在**同一个对象实例**上排队。新对象彼此不共享,等于每个线程锁的都是“自己的门”,根本互斥不了。

常见错误现象:
— 多线程修改共享字段仍出现竞态(如计数器不准)
— 单元测试偶尔通过、压测必崩
— 看似加了锁,实际等效于没加

lock(static_object) 是正确做法,但要注意初始化时机

静态字段保证生命周期和可见性,但必须确保它在首次访问前已就绪。推荐显式声明并初始化,避免依赖静态构造函数的隐式行为。

private static readonly object _syncLock = new object();

// ✅ 安全:字段只读 + 显式初始化
public void DoWork()
{
    lock (_syncLock)
    {
        // 临界区
    }
}

lock(this) 和 lock(typeof(XXX)) 的典型误用

这两类写法看似“复用同一对象”,实则隐患明显,常被当成 static object 的替代方案,但语义和风险完全不同。

真正需要区分的是锁的粒度,而不是“new 还是 static”

关键不在关键字,而在你锁的对象是否对应你要保护的资源范围。比如:

很多人卡在“怎么选锁对象”,其实更该先问:“我要保护什么?谁会并发访问它?最小必要范围是多大?”——锁对象只是这个判断的结果,不是起点。