Java单例模式详解

实现单例,从加载方式来看,有两种:

  • 预加载
  • 懒加载

先看一下实现单例最简单的方式(预加载):

再来看一下懒加载的方式:

以上方式在单线程的情况可以很好的满足需要,换言之,若是在多线程,还需要作一定的改进,如下所示:

上述代码运用了 Double-Checked Locking idiom

解决了多线程环境下的单例,可以进一步思考如何实现可序列化的单例 ? 反序列化可以不通过构造函数直接生成一个对象,所以反序列化时,我们需要保证其不再创建新的对象。

readResolve 方法可以保证,即使程序在上一次运行时序列化过此单例,也只会返回全局唯一的单例。对于 Java 对象序列化机制,可参考附录拓展

java 创建单例的方法基本实现了,不过我们还可以作进一步的改进 —— 代码重构:

好了,现在已经很完美实现了单例的创建,是不是很高兴。单例实线的基本原理,我们已经基本清楚里,最后提供一种更加简洁方法,如下:

08 年 google 开发者年会中,Joshua Bloch Joshua Bloch 在 高效 Java 话题中 解释了这种方法,视频请戳 这里.在 他演讲的ppt30-32 页提到:

高效 Java 线上部分 有说到:

上述实现单例的方式,其实等同于,将 INSTANCE 设置为 public static final 的方式,不同之处在于,使用枚举的方式显得更为简洁,且默认提供了序列化机制,也保证了多线程访问的安全。虽然这种单例的实现方式还未被广泛使用,可实现单例的最好方式就是使用一个单元素的枚举。

为什么可以这么简洁?因为 Java 中每一个枚举类型都默认继承了 java.lang.Enum ,而 Enum 实现了 Serializable 接口,所以枚举类型对象都是默认可以被序列化的。通过反编译,也可以知道枚举常量本质上就是一个

对于枚举的进一步理解,请参考附录拓展

附录拓展: 深入理解 Java 对象序列化 对象的序列化和反序列化 通过反编译字节码来理解 Java 枚举

2 Likes
你目前的身份是游客,评论请输入昵称和电邮!