从单例模式到序列化
单例模式大家应该都不陌生,只要写过一些代码的,估计天天用,好处也自然不多说,有些对象创建一个就够了,比如数据库或者网络的连接,创建一个就够了。实现单例模式需要做到:
- 构造方法私有化
- 静态方法返回实例
- 当心多线程会创建多个实例
- 小心反序列化时会创建多个实例
方法一:类初始化的时候就创建
|
|
这种方式简单粗暴,但是确定是用不到的时候也可能初始化。
方法二:使用的时候创建
|
|
这里需要注意多线程的问题,需要将初始化的代码加一个锁,不过这里这种方式是一种粗粒度的加锁方式,效率不是很高,也可以这么写,效果一样:
网上有一种针对这个的优化版本:
这个方法有一个高大上的名词叫双重校验锁,instance被volatile修饰说明instance任何时候拿到的都是最新的值(可见性),所以如果instance不是null(大部分时候)都不需要进入锁,因为锁比较耗时,所以这样是一个优化。
方式三:使用枚举创建
|
|
这也是《java编程思想》里面推荐的单例模式实现方式,实现非常简单。
方法四:内部静态类
|
|
这个方式算是方法一的升级版,由于内部类在外部类被加载的时候不会立即被加载,只有当getInstance被调用时才加载内部类,所以算是对方法一的一个延迟版本。
一开始说到的反序列化是什么回事呢?
我们看一段代码:
这个输出false,也就是说反序列化的时候单例模式失效了,这是为啥?底层原理我看了网上资料说是反射,没有去读源码来深究,想想反射是可以实现的,这也是为啥单例模式会失效了,所以推荐的是枚举模式,枚举就算反射也没法创建多个实例。针对这个问题,我们可以通过在单例的类中实现一个readResolve方法就行,至于原理,我们下回分析序列化的时候可以一起分析一下。