1.懒汉模式
两次检查的意义在于,防止synchronized同步过程中其他线程进行了实例化。
volatile关键字作用
- 是保证此变量对于所有线程的可见性。即当一条线程修改了这个变量的值,新值对于其他线程来说是可以立即得知的。
- 是禁止指令重排序优化。
JVM 会在保证结果正确的前提下进行指令重排序优化。new一个对象大致有3步。
- 分配内存
- 调用构造函数初始化成实例
- 让instance指向分配的内存空间
即如上 3 步可能的顺序为1->2->3 或 1->3->2 。如果顺序是 1->3->2 ,当 3 执行完,2 还未执行时,另一个线程执行到代码 ③ 处,发现instance不为null,直接返回还未初始化好的instance并使用,就会报错。
2.饿汉式
3.使用静态内部类
SingletonHolder 是静态内部类,当外部类Singleton被加载的时候并不会创建任何实例,只有当Singleton.getInstance()被调用的时候,才会创建Singleton实例,这一切由 JVM 天然完成,所以既保证了线程安全,又实现了延迟加载。
4.枚举类
枚举特点:
- 枚举的构造器总是私有的
- 枚举类中的每个值,都是实例
除此之外,枚举还附带了一些额外好处:无偿地提供了序列化机制,还可以防止通过多次反序列化生成多个实例。