单例模式的代码如下所示:
public class Singleton {
private volatile static Singleton uniqueInstance;
private Singleton() {
}
public static Singleton getUniqueInstance() {
//先判断对象是否已经实例过,没有实例化过才进入加锁代码
if (uniqueInstance == null) {
//类对象加锁
synchronized (Singleton.class) {
if (uniqueInstance == null) {
uniqueInstance = new Singleton();
}
}
}
return uniqueInstance;
}
}
注意点:
-
单例对象 uniqueInstance 使用 volatile 进行修饰,禁止进行指令重排。因为在实例化时候需要进行以下几个动作:
- 为对象申请内存空间
- 初始化对象
- 将指针指向该内存空间
如果进行重排之后先进行了 1 和 3,则有可能会返回没有初始化的对象。
-
因为获取单例对象是静态方法,因此 synchronized 需要获取的锁是类的锁。
-
两次判空,因为如果进入同步代码块之后不进行判空的话,可能会重新进行初始化一次。因为在等待锁的时候可能还没有初始化好,获得锁之后已经初始化好了。
volatile 关键字
除了防止 JVM 的指令重排,还有一个重要的作用就是保证变量的可见性。