概念

单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式。定义为:保证一个类仅有一个实例,并提供一个访问它的全局访问点。

单例模式一般体现在类声明中,单例的类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一的对象的方式,可以直接访问,不需要实例化该类的对象。

用途

单例模式有以下两个优点:

在内存里只有一个实例,减少了内存的开销,尤其是频繁的创建和销毁实例(比如网站首页页面缓存)。

避免对资源的多重占用(比如写文件操作)。

有时候,我们在选择使用单例模式的时候,不仅仅考虑到其带来的优点,还有可能是有些场景就必须要单例,尤其当实例存在多个会引起程序逻辑错误的时候。

实现方式

我们知道,一个类的对象的产生是由类构造函数来完成的。如果一个类对外提供了public的构造方法,那么外界就可以任意创建该类的对象。所以,如果想限制对象的产生,一个办法就是将构造函数变为私有的(至少是受保护的),使外面的类不能通过引用来产生对象。同时为了保证类的可用性,就必须提供一个自己的对象以及访问这个对象的静态方法。

饿汉式

下面是一个简单的单例的实现:

//code 1
public class Singleton {//在类内部实例化一个实例private static Singleton instance = new Singleton();//私有的构造函数,外部无法访问private Singleton() {}//对外提供获取实例的静态方法public static Singleton getInstance() {return instance;}
}

使用以下代码测试:

//code2
public class SingletonClient {public static void main(String[] args) {SimpleSingleton simpleSingleton1 = SimpleSingleton.getInstance();SimpleSingleton simpleSingleton2 = SimpleSingleton.getInstance();System.out.println(simpleSingleton1==simpleSingleton2);}
}

输出结果:

true

code 1就是一个简单的单例的实现,这种实现方式我们称之为饿汉式。所谓饿汉。这是个比较形象的比喻。对于一个饿汉来说,他希望他想要用到这个实例的时候就能够立即拿到,而不需要任何等待时间。所以,通过static的静态初始化方式,在该类第一次被加载的时候,就有一个SimpleSingleton的实例被创建出来了。这样就保证在第一次想要使用该对象时,他已经被初始化好了。

同时,由于该实例在类被加载的时候就创建出来了,所以也避免了线程安全问题。

还有一种饿汉模式的变种:

//code 3
public class Singleton2 {//在类内部定义private static Singleton2 instance;static {//实例化该实例instance = new Singleton2();}//私有的构造函数,外部无法访问private Singleton2() {}//对外提供获取实例的静态方法public static Singleton2 getInstance() {return instance;}
}

code 3和code 1其实是一样的,都是在类被加载的时候实例化一个对象。

饿汉式单例,在类被加载的时候对象就会实例化。这也许会造成不必要的消耗,因为有可能这个实例根本就不会被用到。而且,如果这个类被多次加载的话也会造成多次实例化。其实解决这个问题的方式有很多,下面提供两种解决方式,第一种是使用静态内部类的形式。第二种是使用懒汉式。

静态内部类式

先来看通过静态内部类的方式解决上面的问题:

//code 4
public class StaticInnerClassSingleton {//在静态内部类中初始化实例对象private static class SingletonHolder {private static final StaticInnerClassSingleton INSTANCE = new StaticInnerClassSingleton();}//私有的构造方法private StaticInnerClassSingleton() {}//对外提供获取实例的静态方法public static final StaticInnerClassSingleton getInstance() {return SingletonHolder.INSTANCE;}
}

这种方式同样利用了classloder的机制来保证初始化instance时只有一个线程,它跟饿汉式不同的是(很细微的差别):饿汉式是只要Singleton类被装载了,那么instance就会被实例化(没有达到lazy loading效果),而这种方式是Singleton类被装载了,instance不一定被初始化。因为SingletonHolder类没有被主动使用,只有显示通过调用getInstance方法时,才会显示装载SingletonHolder类,从而实例化instance。想象一下,如果实例化instance很消耗资源,我想让他延迟加载,另外一方面,我不希望在Singleton类加载时就实例化,因为我不能确保Singleton类还可能在其他的地方被主动使用从而被加载,那么这个时候实例化instance显然是不合适的。这个时候,这种方式相比饿汉式更加合理。

懒汉式

下面看另外一种在该对象真正被使用的时候才会实例化的单例模式——懒汉模式。

//code 5
public class Singleton {//定义实例private static Singleton instance;//私有构造方法private Singleton(){}//对外提供获取实例的静态方法public static Singleton getInstance() {//在对象被使用的时候才实例化if (instance == null) {instance = new Singleton();}return instance;}
}

上面这种单例叫做懒汉式单例。懒汉,就是不会提前把实例创建出来,将类对自己的实例化延迟到第一次被引用的时候。getInstance方法的作用是希望该对象在第一次被使用的时候被new出来。

有没有发现,其实code 5这种懒汉式单例其实还存在一个问题,那就是线程安全问题。在多线程情况下,有可能两个线程同时进入if语句中,这样,在两个线程都从if中退出的时候就创建了两个不一样的对象。

线程安全的懒汉式

针对线程不安全的懒汉式的单例,其实解决方式很简单,就是给创建对象的步骤加锁:

//code 6
public class SynchronizedSingleton {//定义实例private static SynchronizedSingleton instance;//私有构造方法private SynchronizedSingleton(){}//对外提供获取实例的静态方法,对该方法加锁public static synchronized SynchronizedSingleton getInstance() {//在对象被使用的时候才实例化if (instance == null) {instance = new SynchronizedSingleton();}return instance;}
}

这种写法能够在多线程中很好的工作,而且看起来它也具备很好的延迟加载,但是,遗憾的是,他效率很低,因为99%情况下不需要同步。(因为上面的synchronized的加锁范围是整个方法,该方法的所有操作都是同步进行的,但是对于非第一次创建对象的情况,也就是没有进入if语句中的情况,根本不需要同步操作,可以直接返回instance。)

双重校验锁

针对上面code 6存在的问题,相信对并发编程了解的同学都知道如何解决。其实上面的代码存在的问题主要是锁的范围太大了。只要缩小锁的范围就可以了。那么如何缩小锁的范围呢?相比于同步方法,同步代码块的加锁范围更小。code 6可以改造成:

//code 7
public class Singleton {private static Singleton singleton;private Singleton() {}public static Singleton getSingleton() {if (singleton == null) {synchronized (Singleton.class) {if (singleton == null) {singleton = new Singleton();}}}return singleton;}
}

code 7是对于code 6的一种改进写法,通过使用同步代码块的方式减小了锁的范围。这样可以大大提高效率。(对于已经存在singleton的情况,无须同步,直接return)。

但是,事情这的有这么容易吗?上面的代码看上去好像是没有任何问题。实现了惰性初始化,解决了同步问题,还减小了锁的范围,提高了效率。但是,该代码还存在隐患。隐患的原因主要和Java内存模型有关。考虑下面的事件序列:

线程A发现变量没有被初始化, 然后它获取锁并开始变量的初始化。

由于某些编程语言的语义,编译器生成的代码允许在线程A执行完变量的初始化之前,更新变量并将其指向部分初始化的对象。

线程B发现共享变量已经被初始化,并返回变量。由于线程B确信变量已被初始化,它没有获取锁。如果在A完成初始化之前共享变量对B可见(这是由于A没有完成初始化或者因为一些初始化的值还没有穿过B使用的内存(缓存一致性)),程序很可能会崩溃。

(上面的例子不太能理解的同学,请恶补JAVA内存模型相关知识)

在J2SE 1.4或更早的版本中使用双重检查锁有潜在的危险,有时会正常工作(区分正确实现和有小问题的实现是很困难的。取决于编译器,线程的调度和其他并发系统活动,不正确的实现双重检查锁导致的异常结果可能会间歇性出现。重现异常是十分困难的。) 在J2SE 5.0中,这一问题被修正了。volatile关键字保证多个线程可以正确处理单件实例

所以,针对code 7 ,可以有code 8 和code 9两种替代方案:

使用volatile

//code 8
public class VolatileSingleton {private static volatile VolatileSingleton singleton;private VolatileSingleton() {}public static VolatileSingleton getSingleton() {if (singleton == null) {synchronized (VolatileSingleton.class) {if (singleton == null) {singleton = new VolatileSingleton();}}}return singleton;}
}

上面这种双重校验锁的方式用的比较广泛,他解决了前面提到的所有问题。但是,即使是这种看上去完美无缺的方式也可能存在问题,那就是遇到序列化的时候。详细内容后文介绍。

使用final

//code 9
class FinalWrapper<T> {public final T value;public FinalWrapper(T value) {this.value = value;}
}public class FinalSingleton {private FinalWrapper<FinalSingleton> helperWrapper = null;public FinalSingleton getHelper() {FinalWrapper<FinalSingleton> wrapper = helperWrapper;if (wrapper == null) {synchronized (this) {if (helperWrapper == null) {helperWrapper = new FinalWrapper<FinalSingleton>(new FinalSingleton());}wrapper = helperWrapper;}}return wrapper.value;}
}

枚举式

在1.5之前,实现单例一般只有以上几种办法,在1.5之后,还有另外一种实现单例的方式,那就是使用枚举:

// code 10
public enum  Singleton {INSTANCE;Singleton() {}
}

这种方式是Effective Java作者Josh Bloch 提倡的方式,它不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象(下面会介绍),可谓是很坚强的壁垒啊,不过,个人认为由于1.5中才加入enum特性,用这种方式写不免让人感觉生疏,在实际工作中,我也很少看见有人这么写过,但是不代表他不好。

单例与序列化

序列化可以破坏单例。要想防止序列化对单例的破坏,只要在Singleton类中定义readResolve就可以解决该问题:

//code 11
import java.io.Serializable;
/*** 使用双重校验锁方式实现单例*/
public class Singleton implements Serializable{private volatile static Singleton singleton;private Singleton (){}public static Singleton getSingleton() {if (singleton == null) {synchronized (Singleton.class) {if (singleton == null) {singleton = new Singleton();}}}return singleton;}private Object readResolve() {return singleton;}
}

总结

本文中介绍了几种实现单例的方法,主要包括饿汉、懒汉、使用静态内部类、双重校验锁、枚举等。还介绍了如何防止序列化破坏类的单例性。

从单例的实现中,我们可以发现,一个简单的单例模式就能涉及到这么多知识。在不断完善的过程中可以了解并运用到更多的知识。所谓学无止境。

不使用synchronized和lock,如何实现一个线程安全的单例?

饿汉模式:使用static来定义静态成员变量或静态代码,借助Class的类加载机制实现线程安全单例。

静态内部类:借助了类加载的时候初始化单例。即借助了ClassLoader的线程安全机制。

枚举:借助了类加载的时候初始化单例。即借助了ClassLoader的线程安全机制。

所谓ClassLoader的线程安全机制,就是ClassLoader的loadClass方法在加载类的时候使用了synchronized关键字。也正是因为这样, 除非被重写,这个方法默认在整个装载过程中都是同步的,也就是保证了线程安全。

所以,以上各种方法,虽然并没有显示的使用synchronized,但是还是其底层实现原理还是用到了synchronized。

还可以使用Java并发包中的Lock实现:

还可以用CAS:CAS是项乐观锁技术,当多个线程尝试使用CAS同时更新同一个变量时,只有其中一个线程能更新变量的值,而其它线程都失败,失败的线程并不会被挂起,而是被告知这次竞争中失败,并可以再次尝试。实现单例的方式如下:

用CAS的好处在于不需要使用传统的锁机制来保证线程安全,CAS是一种基于忙等待的算法,依赖底层硬件的实现,相对于锁它没有线程切换和阻塞的额外消耗,可以支持较大的并行度。

CAS的一个重要缺点在于如果忙等待一直执行不成功(一直在死循环中),会对CPU造成较大的执行开销。

另外,如果N个线程同时执行到singleton = new Singleton();的时候,会有大量对象创建,很可能导致内存溢出。

参考:https://www.jianshu.com/p/fdc64df67794(补充看)

设计模式(一)--单例模式相关推荐

  1. socket可以写成单例嘛_精读《设计模式 - Singleton 单例模式》

    Singleton(单例模式) Singleton(单例模式)属于创建型模式,提供一种对象获取方式,保证在一定范围内是唯一的. 意图:保证一个类仅有一个实例,并提供一个访问它的全局访问点. 其实单例模 ...

  2. C#设计模式(1)——单例模式

    原文地址:http://www.cnblogs.com/zhili/p/SingletonPatterm.html 一.引言 最近在设计模式的一些内容,主要的参考书籍是<Head First 设 ...

  3. java设计模式之单例模式(七种方法)

    单例模式:个人认为这个是最简单的一种设计模式,而且也是在我们开发中最常用的一个设计模式. 单例模式的意思就是只有一个实例.单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例.这个 ...

  4. java单例模式的七种写法_Java设计模式之单例模式的七种写法

    什么是单例模式? 单例模式是一种常见的设计模式,单例模式的写法有很多种,这里主要介绍三种: 懒汉式单例模式.饿汉式单例模式.登记式单例 . 单例模式有以下特点: 1.单例类只能有一个实例. 2.单例类 ...

  5. go设计模式之单例模式

    在这里插入图片描述# go设计模式之单例模式 在软件开发时,经常会遇到有些对象需要保证只有一个实例的,那么这种设计模式就应用而生. 定义 单例模式,也叫单子模式,是一种常用的软件设计模式,属于创建型模 ...

  6. 一篇博客读懂设计模式之---单例模式

    一篇博客读懂设计模式之---单例模式 一.  单例模式 单例对象(Singleton)是一种常用的设计模式.在Java应用中,单例对象能保证在一个JVM中,该对象只有一个实例存在.这样的模式有几个好处 ...

  7. Java 设计模式(3)单例模式

    前言 概念: java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍三种:懒汉式单例.饿汉式单例.登记式单例. 单例模式有以下特点: 1.单例类只能有一个实例. 2.单例类必须自 ...

  8. socket可以写成单例嘛_精读设计模式 Singleton 单例模式

    Singleton(单例模式) Singleton(单例模式)属于创建型模式,提供一种对象获取方式,保证在一定范围内是唯一的. 意图:保证一个类仅有一个实例,并提供一个访问它的全局访问点. 其实单例模 ...

  9. 设计模式之单例模式——Singleton

                        设计模式之单例模式--Singleton 设计意图: 保证类仅有一个实例,并且可以供应用程序全局使用.为了保证这一点,就需要这个类自己创建自己的对象,并且对外有 ...

  10. 实践GoF的设计模式:单例模式

    本文分享自华为云社区<[Go实现]实践GoF的23种设计模式:单例模式>,作者: 元闰子. 简述 GoF 对单例模式(Singleton)的定义如下: Ensure a class onl ...

最新文章

  1. 一场实验室意外爆炸事故,解决了58年量子难题,让科学家意外发现“核电共振”...
  2. 把LabelImg标注的YOLO格式标签转化为VOC格式标签 和 把VOC格式标签转化为YOLO格式标签
  3. iOS组件化(二):创建远程私有库
  4. 【翻译】Programming Ruby——数组,哈希表和控制结构
  5. Spring Cloud微服务系列文,服务调用框架Feign
  6. Linux系统编程7:入门篇之Linux项目自动化构建工具-Make/Makefile的超强使用指南
  7. 37 SD配置-销售凭证设置-分配项目类别
  8. 将xml转为txt_HZ文章转短视频工具v1.0 快速将文章转为短视频 自动配音 配字幕 配图...
  9. python自动化办公实例-python自动化测试实例解析
  10. AD19快速制作多管脚元件符号
  11. 程序猿给娃取名的正确姿势
  12. Xcode 4.3.2 gives error “cannot use super because it is a root class”
  13. 现代诗一首 怀念。。。
  14. 推进个人信息保护的几点建议
  15. mysql 校对规则作用_讲讲Mysql中的校对规则究竟是怎么一回事
  16. 如何表示Unicode的字符?
  17. 数字图像处理习题(一)
  18. 微信小程序 修改button为圆形按钮并设置图片
  19. 数学之美-【算法】 - 用来流方式计算UV的基数算法
  20. Mybatis order by 动态传参出现的一个小bug

热门文章

  1. MySql数据库——文件
  2. 访问我的路径C:\Users\14575\Documents\Tencent Files\1457518657\FileRecv\face\face
  3. L2-009 抢红包 (25分)
  4. 浅谈python高级数据结构—— 字符串(str)
  5. vSphere Web Client 添加主机进VSAN集群时“SAN 主机移至目标群集: vSAN 群集的 UUID 不匹配”报错
  6. C# EF动态获取连接字符串的MSDTC配置
  7. Yoga 14s 2021 Archlinux安装与桌面美化
  8. uCosII移植STM32F407教程
  9. vue实现类目筛选功能
  10. 从SVN上下载项目到启动项目