public class Singleton {
private Singleton() {} //关键点0:构造函数是私有的
private static Singleton single = null; //关键点1:声明单例对象是静态的
private static object obj= new object();
public static Singleton GetInstance() //通过静态方法来构造对象
{
if (single == null) //关键点2:判断单例对象是否已经被构造
{
lock(obj) //关键点3:加线程锁
{
if(single == null) //关键点4:二次判断单例是否已经被构造
{
single = new Singleton();
}
}
}
return single;
}
}
在判断单例实例是否被构造时,需要检测两次,在线程锁之前判断一次,在线程锁之后判断一次,再去构造实例,这样就万无一失了。
为什么需要判断两次?
如遇到延迟加载或者缓存原因,可能造成构造多个实例,违反了单例的初衷。所以需要两次。
单例类的构造函数必须私有化,单例类不能被实例化,单例实例只能静态调用。
lock锁住的必须是object对象,不能是int。锁住的必须是引用类型。如果锁值类型,每个不同的线程在声明的时候值类型变量的地址都不一样,那么上个线程锁住的东西下个线程进来会认为根本没锁,相当于每次都锁了不同的门,这样并没有什么意义。而引用类型的变量地址是相同的,每个线程进来判断锁多想是否被锁的时候都是判断同一个地址,相当于是锁在通一扇门,起到了锁的作用。
所以单例是为了保证系统中只有一个实例,其关键点如下:
一.私有构造函数
二.声明静态单例对象
三.构造单例对象之前要加锁(lock一个静态的object对象)
四.需要两次检测单例实例是否已经被构造,分别在锁之前和锁之后