1、单例模式的作用
节省内存和计算、保证结果的正确、方便管理
2、单例模式的适用场景
(1)无状态的工具类,如日志打印等
(2)全局的信息类,只需要一个全局共享的信息类,便于管理保证数据的一致性
3、实现单例模式的7种写法1:饿汉式(静态常量)(线程安全,非懒加载):
在由static修饰的INSTANCE在类加载的时候就完成了实例的创建,如果不使用它可能存在内存浪费的情况,但它是线程安全的。
4、实现单例模式的7种写法2:饿汉式(静态代码块)(线程安全,非懒加载)
效果与静态常量相同
5、实现单例模式的7种写法3:懒汉式(线程不安全)(线程不安全,懒加载)
懒加载:不在类加载的时候创建对象,等到了真正使用的时候才去创建
它是线程不安全的,开始时两个线程同时运行可能返回两个不同的对象
如图在对象创建之前加上锁也不是线程安全的。
a、instance=new Singleton3()不是一个原子 *** 作,它包含了3个步骤:创建一个对象实例,将对象初始化,将instance指向对象实例。后两步是有可能发生重排序的。
b、Synchronized不能防止重排序
c、异常情况1:当线程1运行至16行时如果发生了重排序,先让instance指向对象,然后做初 始化;而线程2运行到了第14行,此时instance不为空,直接返回对象,但是此时对象还没有初始化,就会导致拿到的对象是错误的
d、异常情况2:线程1和2同时运行到第15行,然后两个线程会先后创建两个不同的对象并返回。
6、实现单例模式的7种写法4:懒汉式(线程安全)(线程安全,效率低,懒加载)
在getInstance()方法前加上锁,可以保证线程安全但是效率低
7、实现单例模式的7种写法5:双重检查(线程安全,懒加载)
在实现懒加载的同时,解决线程不安全的问题:
a、将instance设置成volatile是解决5中所讲的异常情况1,防止运行instance=new Single5()时发生重排序
b、13行的第一层检查是为了提高效率
c、15行的第二层检查是解决5中所讲的异常情况2,防止两次个线程拿到不同的两个对象。
8、实现单例模式的7种写法6:静态内部类(线程安全,懒加载)
利用JVM在加载外部类时不会实例化内部类中的对象的特点,保证了懒加载
9、实现单例模式的7种写法7:枚举(线程安全,懒加载)
推荐使用的方法,调用直接使用,Singleton7.INSTANCE.whatever();
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)