1.懒汉式:
方法一:同步延迟加载 — synchronized方法
运行效率会很低,因为同步块的作用域有点大,而且锁的粒度有点粗。同步方法效率低。
方法二:
另外,需要注意 uniqueInstance 采用 volatile 关键字修饰也是很有必要。
uniqueInstance 采用 volatile 关键字修饰也是很有必要的, uniqueInstance = new Singleton(); 这段代码其实是分为三步执行:
- 为 uniqueInstance 分配内存空间
- 初始化 uniqueInstance
- 将 uniqueInstance 指向分配的内存地址
但是由于 JVM 具有指令重排的特性,执行顺序有可能变成 1->3->2。指令重排在单线程环境下不会出现问题,但是在多线程环境下会导致一个线程获得还没有初始化的实例。例如,线程 T1 执行了 1 和 3,此时 T2 调用 getUniqueInstance() 后发现 uniqueInstance 不为空,因此返回 uniqueInstance,但此时 uniqueInstance 还未被初始化。
使用 volatile 可以禁止 JVM 的指令重排,保证在多线程环境下也能正常运行。
2.饿汉式:
或者使用饿汉写法:
饿汉式天生线程安全,可以直接用于多线程而不会出现问题。
class Singleton0 {
//构造方法私有化
private Singleton0() {
}
//内部创建静态对象并实例化
private static Singleton0 s=new Singleton0();
//提供静态方法返回该静态对象
public static Singleton0 getsingle() {
return s;
}
}