死磕设计模式—单例模式
1.1.简介
例如在Java
开发中,我们都知道类和对象实例可以通过new
来创建一个或者多个,而单例模式就是采取一定的办法保证整个系统中某一个类只能存在唯一一个对象实例,并且获取该类实例的方法只能是该类自己提供的一个获取其实例的静态方法。
1.2.使用样例
例如在Spring
源码中的doGetBean
方法中就使用到了单例模式,如下:
或者JDK
中的RunTime
类就使用到了饿汉式的实现方式创建单例,如下:
[图片上传失败...(image-c02ff2-1564018415917)].png)
单例模式认知小结:
1.一个类只有一个实例;2.类的实例只能是由自己创建;3.该类必须向外提供获取自己实例的静态方法;
1.2.应用场景
在《图解设计模式》一书中详细介绍了单例模式在开发中的使用场景,如下所示:
一个系统中可以存在多个打印任务,但是只能有一个正在工作的任务;一个系统只能有一个窗口管理器或文件系统;一个系统只能有一个计时工具或ID(序号)生成器
1.2.代码实现
饿汉式(静态变量)
- 类构造器私有外(防止外部使用可以
new
一个实例) - 类内部定义一个静态变量,在类的内部创建该实例
- 提供一个静态的公共方法返回该类的实例
public class Singleton {
private Singleton() {}
private static final Singleton singleton = new Singleton();
public static Singleton getInstance() {
return singleton;
}
}
优点:代码实现简单,在类加载的时候就完成了类的实例化,避免了线程同步问题。
缺点:在类加载的时候完成了实例化,没有达到懒加载的效果,可能不使用到这个实例,就会造成内存浪费。
饿汉式(静态代码块)
public class Singleton {
private Singleton() {}
private static final Singleton singleton;
static {
singleton = new Singleton();
}
public static Singleton getInstance() {
return singleton;
}
}
使用静态代码块来完成类的实例化和使用静态变量的优缺点是一样的。
懒汉式(线程不安全)
public class Singleton {
private Singleton() {}
private static Singleton singleton;
public static Singleton getInstance() {
if (singleton == null) {
return new Singleton();
}
return singleton;
}
}
注意:这种实现方式不推荐使用,虽然达到了懒加载的效果,但是只能在单线程下使用,在多线程下存在线程安全问题。
懒汉式(线程安全,同步方法)
public class Singleton {
private Singleton() {}
private static Singleton singleton;
public static synchronized Singleton getInstance() {
if (singleton == null) {
return new Singleton();
}
return singleton;
}
}
注意:这种实现方法也不推荐使用,虽然解决了线程安全问题,但是使用了
synchronized
这个庞大的东西,使得效率太低太低,每次执行getInstance()
方法,都要进行方法同步。
懒汉式(线程不安全,同步代码块)
public class Singleton {
private Singleton() { }
private static Singleton singleton;
public static Singleton getInstance() {
if (singleton == null) {
synchronized (Singleton.class) {
return new Singleton();
}
}
return singleton;
}
}
注意:同步代码块的实现方式是在同步方法上改进的,因为使用同步方法使得效率很低,改为同步代码块,但是这种实现方法并不能解决线程安全问题,所以,在实际开发中不推荐使用。
双重检查(Double-check,推荐使用)
public class Singleton {
private Singleton() { }
private static volatile Singleton singleton = null;
public static Singleton getInstance() {
if (singleton == null) {
synchronized (Singleton.class) {
if (singleton == null) {
return new Singleton();
}
}
}
return singleton;
}
}
说明:线程安全、延迟加载、效率较高 、推荐使用
实例化代码只用执行一次,后面再次访问时,判断if (singleton == null), 直接return实例化对象,也避免的反复进行方法同步 。
静态内部类(线程安全,推荐使用)
public class Singleton {
private Singleton() { }
private static class SingletonInstance {
private static final Singleton INSTANCE = new Singleton();
}
public static Singleton getInstance() {
return SingletonInstance.INSTANCE;
}
}
- 这种方式采用了类装载的机制来保证初始化实例时只有一个线程。
- 静态内部类方式在
Singleton
类被装载时并不会立即实例化,而是在需要实例化 时,调用getInstance
方法,才会装载SingletonInstance
类,从而完成Singleton
的 实例化。 - 类的静态属性只会在第一次加载类的时候初始化,所以在这里,
JVM
帮助我们保证了线程的安全性,在类进行初始化时,别的线程是无法进入的。 - 避免了线程不安全,利用静态内部类特点实现延迟加载,效率高 ,推荐使用。
枚举(推荐使用)
public enum SingletonEnum {
INSTANCE;
public void method() {
}
}
SingletonEnum instance = SingletonEnum.INSTANCE;
说明:不仅能避免多线程同步问题,而 且还能防止反序列化重新创建新的对象 ,缺点:不能懒加载。
1.3.参考文献
更多文章,更好的阅读体验,请前往个人网站查看 码酱博客