一、描述:
二、主要特点:
三、单例模式的应用:
四、实现方法:
1、饿汉式
应用场景:如果应用程序总是创建并使用单例实例或在创建和运行时开销不大
- public class EagerSingleton {
- // jvm保证在任何线程访问uniqueInstance静态变量之前一定先创建了此实例
- private static EagerSingleton uniqueInstance = new EagerSingleton();
- // 私有的默认构造子,保证外界无法直接实例化
- private EagerSingleton() {
- }
- // 提供全局访问点获取唯一的实例
- public static EagerSingleton getInstance() {
- return uniqueInstance;
- }
- }
2、懒汉式
应用场景:如果开销比较大,希望用到时才创建就要考虑延迟实例化,或者Singleton的初始化需要某些外部资源(比如网络或存储设备)。
- public class LazySingleton {
- private static LazySingleton uniqueInstance;
- private LazySingleton() {
- }
- public static synchronized LazySingleton getInstance() {
- if (uniqueInstance == null)
- uniqueInstance = new LazySingleton();
- return uniqueInstance;
- }
- }
3、"双检锁"(Double-Checked Lock)
同步一个方法可能造成程序执行效率下降100倍,完全没有必要每次调用getInstance都加锁,事实上我们只想保证一次初始化成功,其余的快速返回而已,如果在getInstance频繁使用的地方就要考虑重新优化,"双检锁"(Double-Checked Lock)尽量将"加锁"推迟,只在需要时"加锁"(仅适用于Java 5.0 以上版本,volatile保证原子操作)。
volatile 的功能:
1. 避免编译器将变量缓存在寄存器里
2. 避免编译器调整代码执行的顺序
1. 避免编译器将变量缓存在寄存器里
2. 避免编译器调整代码执行的顺序
- public class DoubleCheckedLockingSingleton {
- // java中使用双重检查锁定机制,由于Java编译器和JIT的优化的原因系统无法保证我们期望的执行次序。
- // 在java5.0修改了内存模型,使用volatile声明的变量可以强制屏蔽编译器和JIT的优化工作
- private volatile static DoubleCheckedLockingSingleton uniqueInstance;
- private DoubleCheckedLockingSingleton() {
- }
- public static DoubleCheckedLockingSingleton getInstance() {
- if (uniqueInstance == null) {
- synchronized (DoubleCheckedLockingSingleton.class) {
- if (uniqueInstance == null) {
- uniqueInstance = new DoubleCheckedLockingSingleton();
- }
- }
- }
- return uniqueInstance;
- }
- }
4、内部类实现
- public class LazyInitHolderSingleton {
- private LazyInitHolderSingleton() {
- }
- private static class SingletonHolder {
- private static final LazyInitHolderSingleton INSTANCE = new LazyInitHolderSingleton();
- }
- public static LazyInitHolderSingleton getInstance() {
- return SingletonHolder.INSTANCE;
- }
- }
5、枚举类型实现
- public class SingletonClass{
- INSTANCE;
- }