先举典型的例子,单例模式。
View Code
1 public sealed class Singleton
2 {
3 private Singleton(){}
4 private static Singleton instance = null;
5 private static object syncRoot = new object();
6
7 public static Singleton Instance
8 {
9 get
10 {
11 if (instance == null)
12 {
13 lock (syncRoot)
14 {
15 if (instance == null)
16 {
17 instance = new Singleton();
18 }
19 }
20 }
21 return instance;
22 }
23 }
24 }
这里简单说一下,lock(syncRoot) 获取对象syncRoot的互斥锁,可以简单理解为,当多个线程同时执行到lock的时候,大家排队,一个一个地进行。C#中的lock对应于Java中的synchronized。这里在11行与15行进行了重复检查,有些人认为是没有必要的。因为下面的代码是等效的。
View Code
1 public sealed class Singleton
2 {
3 private Singleton(){}
4 private static Singleton instance = null;
5 private static object syncRoot = new object();
6
7 public static Singleton Instance
8 {
9 get
10 {
11 lock (syncRoot)
12 {
13 if (instance == null)
14 {
15 instance = new Singleton();
16 }
17 }
18 return instance;
19 }
20 }
21 }
那么我们就结合实际的情况来分析一下二者的区别,为了说明方便,以上两种情况分别简称为 “双重检查锁”和“单重检查锁”。
1、第一次访问Instance,同时来了10个线程。对于双重检查锁,instance为null,10个线程在这里lock处排队;对于单重检查锁,10个线程在lock处排队。二者是相同的。
2、第二次、第三次。。。访问Instance,同时来了10个线程。对于双重检查锁,instance不为null,10个线程不用排队,直接返回instance;对于单重检查锁,10个线程,还必须要在lock处排队。
双重检查锁的优点体现出来了:避免了不必要的排队现象。也就是说,双重检查锁的第一重检查,是很必要的,它来保证不必要的排队。
举个例子说明,病人到医院看病,第一次去的时候,都要排队去办病历本。以后再去的时候,如果有病历本,就不用再排队去办了。