各种单例模式问题分析演变

单例模式

构造器私有化 通过方法调用返回实例
在任何情况下都只有一个实例
show code见上篇撸代码:https://www.jianshu.com/p/1f5032a27a03

1.饿汉式

静态 直接赋值
优点:执行效率高,性能高,没有锁
缺点:启动就加载,可能内存浪费;启动慢

2.懒汉式

2.1简单懒汉式

延时加载
延时加载因为临时调用赋值对象,需要判空
优点:节省内存
缺点:线程不安全,多个线程可能同时调用创建方法,同时进入方法,同时判空,会创建多个实例

2.2加锁

调用方法加synchronized 控制资源占用
缺点:阻塞线程成为瓶颈,性能大受影响

2.3双重检查

减小锁粒度,不加在方法中,加在代码块中,先让线程进入方法,通过一层判空先过滤掉一些线程
第一次检查是否需要阻塞,减少阻塞
第二次检查是否需要重新创建实例,避免多实例
避免指令重排序问题,实例变量需要加volatile
缺点:不优雅,

2.4静态内部类

内部类中直接创建,不是饿汉式;
类加载时加载外部类 outer.class,内部类不会加载;
优点:写法优雅,利用java本身语法特点,性能高,避免内存浪费
缺点:能被反射攻击,
以上所有写法都存在这个问题,私有构造器可以通过反射暴力授权破解
可以通过构造器判空解决
缺点:不优雅,构造器私有不创建,写判断逻辑可读性差

3.注册式单例

java effective:将每一个实例都缓存到统一的容器中,使用唯一标识获取实例

3.1枚举式单例

enum没有无参构造方法,有一个string,int参数的构造器;
enum不能通过反射创建枚举实例
通过官方jdk底层保证单例
enum值由一个map保存,启动时创建了,类似饿汉式,没有线程安全问题;enum可以有多个key;没有反序列化问题

缺点:不能大批量创建,浪费内存;

3.2ioc容器

spring单例容器
大批量创建对象
托管容器来管理bean,创造了一个生态,在此生态内保证单例,如果单独调用bean会有反射问题
需要解决线程安全问题;有反序列化问题,以上所有写法均有此问题

反序列化问题解决
类中增加readResolve(),返回对象;利用桥接模式
原理:
在反序列化readObject中,有readResolve()判断不会重新创建

4.ThreadLocal单例

线程内部全局唯一,天生线程安全
ThreadLocal<class本身>
同一个线程内唯一,

5.源码单例运用

AbstractFactoryBean
mybatis:ErrorContext threadLocal单例应用

6.单例总结

优点:内存中一个实例,减少内存占用;唯一访问点
缺点:没有接口,扩展困难
重点:
1.私有构造器
2.线程安全
3.延迟加载
4.防止反序列化,反射攻击

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容