C#中的线程安全是什么 C# lock关键字和Monitor实现线程同步

来源:这里教程网 时间:2026-02-21 17:31:09 作者:

在多线程编程中,多个线程同时访问共享资源可能会导致数据不一致或程序行为异常。C#中的线程安全指的是在多线程环境下,某个方法、类或代码块能够正确地处理多个线程的并发访问,不会产生竞态条件(Race Condition)或数据损坏。

为了实现线程安全,C#提供了多种同步机制,其中最常用的是 lock 关键字和 Monitor 类。它们都基于对象的互斥锁(Mutex)机制,确保同一时间只有一个线程可以进入临界区。

lock关键字:简洁的同步语法

lock 是 C# 中用于实现线程同步的语法糖,它确保一段代码在同一时刻只能被一个线程执行。

使用 lock 时,需要指定一个对象作为“锁对象”,该对象用于控制对临界区的访问。

private static readonly object lockObj = new object();
private static int counter = 0;
<p>public static void Increment()
{
lock (lockObj)
{
counter++;
Console.WriteLine($"当前计数: {counter}");
}
}</p>

上面的代码中,lock 保证了 counter++ 和输出操作作为一个原子操作执行,避免多个线程同时修改 counter 导致数据错误。

注意:锁对象应为私有的、静态的 object 实例,避免使用 public 或 this,防止外部锁定造成死锁或安全问题。

Monitor类:lock背后的实现机制

lock 关键字实际上是 Monitor 类的简化封装。编译器会将 lock 块翻译成对 Monitor.Enter 和 Monitor.Exit 的调用。

Monitor.Enter(lockObj);
try
{
    counter++;
    Console.WriteLine($"当前计数: {counter}");
}
finally
{
    Monitor.Exit(lockObj);
}

这种结构确保即使发生异常,也能正确释放锁,避免死锁。

Monitor 还提供了一些更灵活的方法:

Monitor.TryEnter:尝试获取锁,可设置超时时间,避免无限等待。 Monitor.Wait:释放锁并等待其他线程调用 Notify。 Monitor.Pulse / PulseAll:唤醒等待中的线程。

这些方法可用于实现生产者-消费者等复杂同步场景。

lock与Monitor的选择

在大多数情况下,推荐使用 lock,因为它语法简洁、不易出错,且由编译器保证 finally 块中释放锁。

只有在需要更精细控制(如超时、线程通信)时,才直接使用 Monitor。

例如,使用 TryEnter 避免死锁:

if (Monitor.TryEnter(lockObj, TimeSpan.FromSeconds(1)))
{
    try
    {
        // 执行临界区操作
    }
    finally
    {
        Monitor.Exit(lockObj);
    }
}
else
{
    Console.WriteLine("无法获取锁,跳过操作");
}

基本上就这些。理解 lock 和 Monitor 的关系,有助于写出更安全、高效的多线程代码。关键是保护好共享状态,避免过度锁定影响性能。

相关推荐

热文推荐