创建单例模型的方法有多种,我们常用的是双重校验法,代码如下:
public class SingleTon {
private static SingleTon instance = null;
private SingleTon(){}
public static SingleTon getInstance(){
if(instance == null) {//第一次判断是否为null
synchronized (SingleTon.class){//使用synchronized对class加锁
if(instance == null) {//再次判断是否为空
instance = new SingleTon();
}
}
}
return instance;
}
}
上述代码,在执行10万、20万次可能没有问题,但是总有可能会出现一种报null的异常;分析如下:
instance = new SingleTon();
上面的代码,在底层其实是分成了3步:
1.分配地址空间 2.初始化SingleTon对象 3.将SingleTon对象的地址赋值给instance,此时instance不为null
我们都知道,jvm为了提高性能,编译器和处理器都会对指令进行重排序,上述的3步,中2和3是没有数据依赖,可以重排序,所以可能的执行顺序是1/3/2,所以当第一个线程在按照1/3/2创建单例时,正好执行到了3步骤,只是将地址空间赋值给instance,而没有初始化SingleTon对象,第二个线程过来以判断instance不为null,直接就调用单例内部其他方法的话可能就会报null
总结,所以在使用双重校验创建单例时,instance需要使用volatile来修饰