一、单例模式介绍
1.1 定义
保证一个类仅有一个实例,并提供一个访问它的全局访问点。
1.2 为什么要用单例模式?
在我们的系统中,有一些对象其实我们只需要一个,比如说:线程池、缓存、对话框、注册表、日志对象、充当打印机、显卡等设备驱动程序的对象。事实上,这一类对象只能有一个实例,如果制造出多个实例就可能会导致一些问题的产生,比如:程序的行为异常、资源使用过量、或者不一致性的结果。
简单来说使用单例模式可以带来下面几个好处:
- 对于频繁使用的对象,可以省略创建对象所花费的时间,这对于那些重量级对象而言,是非常可观的一笔系统开销;
- 由于 new 操作的次数减少,因此对系统内存的使用频率也会降低,这将减轻 GC 压力,缩短 GC 停顿时间。
1.3 为什么不使用全局变量确保一个类只有一个实例呢?
我们知道全局变量分为静态变量和实例变量,静态变量也可以保证该类的实例只存在一个。
只要程序加载了类的字节码,不用创建任何实例对象,静态变量就会被分配空间,静态变量就可以被使用了。
但是,如果说这个对象非常消耗资源,而且程序某次的执行中一直没用,这就造成了资源的浪费。利用单例模式的话,我们就可以实现在需要使用时才创建对象,这样就避免了不必要的资源浪费。不仅仅是因为这个原因,在程序中我们要尽量避免全局变量的使用,大量使用全局变量给程序的调试、维护带来困难。
二 单例模式的实现
通常单例模式在 Java 语言中,有两种创建方式:
- 饿汉方式。指全局的单例在类装载时创建
- 懒汉方式。指全局的单例在第一次被使用时构建。
不管是哪种创建方式,它们通常存在下面几点相似处:
- 单例类必须要有一个 private 访问级别的构造函数,只有这样,才能确保单例不会在系统中的其他代码内被实例化;
- instance 成员变量和 uniqueInstance 方法必须是 static 的。
2.1 饿汉方式(线程安全)
public class Singleton { //在静态初始化器中创建单例实例,这段代码保证了线程安全 private static Singleton uniqueInstance = new Singleton(); //Singleton类只有一个构造方法并且是被private修饰的,所以用户无法通过new方法创建该对象实例 private Singleton(){ } public static Singleton getInstance() { return uniqueInstance; } }
所谓饿汉方式就是说 JVM 在加载这个类时就马上创建此唯一的单例实例,不管你用不用,先创建了再说,如果一直没有被使用,便浪费了空间,典型的空间环时间,每次调用的时候,就不需要再判断,节省了运行时间。
2.2 懒汉式(非线程安全和synchronized关键字线程安全版本)
public class Singleton { private static Singleton uniqueInstance; private Singleton() { } //没有加入synchronized关键字的版本时线程不安全的 public static Singleton getInstance() { //判断当前单例是否已经存在,若存在则返回,不存在则再建立单例 if(uniqueInstance == null) { uniqueInstance = new Singleton(); } return uniqueInstance; } }
所谓懒汉式就是说单例实例在第一次被使用时构建,而不是 JVM 在加载这个类时就马上创建此唯一的单例实例。
但是上面这种方式很明显非线程安全的,如果多个线程同时访问getInstance()方法时就会出现问题。如果想要保证线程安全,一种比较常见的方式就是在getInstance()方法前加上synchronized关键字,如下:
public static synchronized Singleton getInstance() { //判断当前单例是否已经存在,若存在则返回,不存在则再建立单例 if(uniqueInstance == null) { uniqueInstance = new Singleton(); } return uniqueInstance; }
我们知道synchronized关键字偏重量级锁。虽然在JavaSE1.6之后synchronized关键字进行了主要包括:为了减少获得锁和释放锁带来的性能消耗而引入的偏向锁和轻量级锁以及其他各种优化之后指向效率有了显著提升。
但是在程序中每次使用getInstance()都要经过synchronized加锁这一层,这难免会增加getInstance()的时间开销,而且还可能发生阻塞。我们下面介绍的双重检查加锁版本是为了解决这个问题。
2.3 懒汉式(双重检查加锁版本)
利用双重检查加锁(double-checked locking),首先检查是否实例已经创建,如果尚未创建,才进行同步。这样一来,只有一次同步,这只是想要的效果。
public class Singleton { private volatile static Singleton uniqueInstance; private Singleton() { } public static Singleton getInstance() { //检查实例,如果不存在,就进入同步代码块 if(uniqueInstance == null) { synchronized (Singleton.class) { //进入同步代码块后,再检查一次,如果仍是null,才创建实例 if(uniqueInstance == null) { uniqueInstance = new Singleton(); } } } return uniqueInstance; } }
很明显,这种方式相比于使用synchronized关键字的方法,可以大大减少getInstance() 的时间消费。
注意: 双重检查加锁版本不适用于1.4及更早版本的Java。
1.4及更早版本的Java中,许多JVM对于volatile关键字的实现会导致双重检查加锁的失效。
2.4 懒汉式(登记式/静态内部类方式)
静态内部实现的单例是懒加载的且线程安全
只有通过显示调用 getInstance 方法时,才会显示装载 SingletonHolder 类,从而实例化 instance (只有第一次使用这个单例的实例的时候才加载,同时不会有线程安全问题)
public class Singleton { private static class SingletonHolder { private static final Singleton INSTANCE = new Singleton(); } private Singleton(){} public static final Singleton getInstance() { return SingletonHolder.INSTANCE; } }
2.5 饿汉式(枚举方式)
这种实现方式还没有被广泛采用,但这是实现单例模式的最佳方法。它更简洁,自动支持序列化机制,绝对防止多次实例化(如果单例类实现了Serializable接口,默认情况下每次反序列化总会创建一个新的实例对象,关于单例与序列化的问题可看《单例与序列化的那些事儿》)
public enum Singleton { //定义一个枚举的元素,它就是 Singleton 的一个实例 INSTANCE; public void doSomeThing() { System.out.println("枚举方法实现单例"); } }
使用方法:
public class Solution { public static void main(String[] args) { Singleton singleton = Singleton.INSTANCE; singleton.doSomeThing(); //output:枚举方法实现单例 } }
这种方法在功能上与公有域方法相近,但是它更加简洁,无偿提供了序列化机制,绝对防止多次实例化,即使是在面对复杂序列化或者反射攻击的时候。虽然这种方法还没有广泛采用,但是单元素的枚举类型已经成为实现Singleton的最佳方法。 —-《Effective Java 中文版 第二版》
2.6 总结
我们主要介绍到了以下几种方式实现单例模式:
- 饿汉方式(线程安全)
- 懒汉式(非线程安全和synchronized关键字线程安全版本)
- 懒汉式(双重检查加锁版本)
- 懒汉式(登记式/静态内部类方式)
- 饿汉式(枚举方式)