先上代码
public class SingletonTest { // 使用volatile保证可见性 private static volatile SingletonTest singletonTest = null; private SingletonTest() { } public static SingletonTest getInstance() { //第一重校验 if (singletonTest == null) { // 加锁 synchronized (SingletonTest.class) { // 第二重校验 if (singletonTest == null) { singletonTest = new SingletonTest(); } } } return singletonTest; } }
这里面有2个关键点
1.使用volatile保证可见性
volatile功能:当一个共享变量被volatile修饰时,它会保证修改的值会立即被更新到主存,当有其他线程需要读取时,它会去内存中读取新值
2.使用双重校验
这里为一直无法理解,为什么要加双重校验锁,我原本以为理论上,只要在synchronized里面加一层校验锁,也不会产生单例多线程问题.后面我才理解,如果加了第一层校验,则除例首次需要加锁外,后续都不会进行加锁操作,因为已经获取到了实例.如果只加一层校验锁,则每次获取都要加锁,无疑会加大了许多开销.