单例模式
饿汉式
public class SingleTon {private static SingleTon instance = new SingleTon();private SingleTon(){};public static SingleTon getInstance(){return instance;}
}
要点:
- 静态 private instance
- 私有的构造器
- 方法返回 instance
懒汉式
public class SingleTon {private volatile static SingleTon instance;private SingleTon(){}public static SingleTon getInstance(){if(instance == null){synchronized (SingleTon.class){if(instance == null){instance = new SingleTon();}}}return instance;}
}
要点:
- volatile 修饰 instance
- 双端锁检验
- 构造器私有化
为什么需要 volatile:
主要由于双端锁检验和指令重排共同造成的问题:
- 双端锁减少了在锁外等待的线程,提高了效率但是也造成了问题。
- new 对象的过程
- 分配空间
- 初始化对象
- 引用指向对象
- 其中第 2,3 步可能会重排,造成 instance 判断不为空,但是并未初始化,结果该线程不再等待锁直接返回了 instance,此时 instance 未被初始化,线程不安全。
- volatile 禁止指令重排,先初始化再引用指向。