Java设计模式 —— 单例模式
6 单例模式
6.1 单例模式概述
Singleton Patter:确保一个类只有一个实例,并提供一个全局访问点来访问这个唯一实例。
单例模式有3个要点:
- 该类只能有一个实例
- 该类必须自行创建这个实例
- 该类必须向整个系统提供这个实例
单例模式结构图如下所示:

6.2 单例模式实现
6.2.1 单例类
public class Singleton {
// 静态私有成员变量
private static Singleton instance = null;
// 私有构造函数, 类外无法创建该类实例
private Singleton() {
}
// 静态公有访问方法
public static Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
6.2.2 客户端调用
public class Client {
public static void main(String[] args) {
Singleton s1 = Singleton.getInstance();
Singleton s2 = Singleton.getInstance();
// 判断两个对象是否是同一实例
if (s1 == s2) {
System.out.println("s1 and s2 are the same.");
}
else {
System.out.println("s1 is different from s2.");
}
}
}
// Result
>>> s1 and s2 are the same.
单例模式的实现过程需要注意以下3点:
- 构造函数私有
- 提供一个类型为自身的静态私有成员变量
- 提供一个公有的静态工厂方法
6.3 饿汉式单例
饿汉式单例类是最简单的单例类,该模式结构图如下:

从图中可以看初,在定义静态变量时实例化单例类,因此在类加载时单例对象就已创建。
public class EagerSingleton {
private static final EagerSingleton instance = new EagerSingleton();
private EagerSingleton() {}
public static EagerSingleton getInstance() {
return instance;
}
}
6.4 懒汉式单例
6.4.1 懒汉式单例概述
与饿汉式单例不同的是,懒汉式单例会在第一次被引用时将自己实例化,在懒汉式单例类被加载时不会实例化。懒汉式单例结构图如下:

懒汉式单例在第一次调用 getInstance() 方法时实例化,在类加载时并不实例化,这种技术称为延迟加载(Lazy load)。
6.4.2 双重检查锁定
Step1:为了避免多个线程同时调用 getInstance() 方法,可以使用 synchronized 关键字
public class LazySingleton {
private static LazySingleton instance = null;
private LazySingleton() {}
synchronized public static LazySingleton getInstance() {
if (instance == null) {
instance = new LazySingleton();
}
return instance;
}
}
Step2:在 Step1 的懒汉式单例中,在 getInstance() 方法前加了关键字进行线程锁定,以处理多个线程访问的问题。但是每次方法调用时都需要进行线程锁定判断,导致性能大大降低。因此可以对上述代码进行改进,无须对整个 geInstance() 方法锁定,只需对代码块 instance = new LazySingleton() 锁定即可。
public class LazySingleton {
private static LazySingleton instance = null;
private LazySingleton() {}
public static LazySingleton getInstance() {
if (instance == null) {
synchronized(LazySingleton.class) {
instance = new LazySingleton();
}
}
return instance;
}
}
Step3:Step2 中代码实现貌似解决了线程安全问题,但事实并非如此,还是会存在创建多个单例对象的情况,原因如下:
假如线程A和线程B都在调用 getInstance() 方法,此时 instance 对象为 null,两个均能通过 if 判断语句。假如线程A拿到了类锁执行实例创建,线程B处于排队等待,当线程A执行完毕后,线程B并不知道实例已经创建,将继续创建实例,导致产生多个单例对象。
进一步修改代码,在 synchronized 中再进行一次判断,这种方式称为双重锁定检查。
public class LazySingleton {
// volatile关键字,确保变量修改对其他线程可见
private volatile static LazySingleton instance = null;
private LazySingleton() {}
public static LazySingleton getInstance() {
// 第一重判断
if (instance == null) {
synchronized(LazySingleton.class) {
// 第二重判断
if (instance == null) {
instance = new LazySingleton();
}
}
}
return instance;
}
}
6.5 静态内部类实现单例模式
饿汉式单例类在类加载时就实例化,不管将来用不用始终占用内存;懒汉式单例类线程安全控制繁琐,性能受到影响;饿汉式和懒汉式单例都存在一些问题,为了克服这些问题,在 Java 中可以通过 Initialization on Demand Holder(IoDH) 技术来实现。
在 IoDH 中,需要在单例类中新增一个静态内部类,在该内部类创建单例对象。
public class Singleton {
private Singleton() {}
private static class HoldClass {
private final static Singleton instance = new Singleton();
}
public static Singleton getInstance() {
return HoldClass.instance;
}
}
由于静态单例对象不是 Singleton 的成员变量,因此在加载 Singleton 类时不会将其实例化,第一次调用 getInstance() 方法时将加载内部类 HoldClass 初始化 instance,由 Java 虚拟机来保证其线程安全性,确保该成员变量只能初始化一次;且由于 getInstance() 方法没有任何线程锁定,因此不会对其性能造成影响。
通过 IoDH 既可以实现延迟加载,又可以保证线程安全,不失为一种最好的 Java 语言单例模式的实现方式,其缺点是与编程语言本身的特性有关。
6.6 单例模式优/缺点
单例模式的优点主要如下:
- 在系统中只存在一个对象,可以节约系统资源
- 单例类封装了它的唯一实例,可以严格控制该实例的访问方式
单例模式的缺点主要如下:
- 单例模式中没有抽象层,扩展困难
- 单例类既提供业务方法,又提供对象创建方法,将对象的创建和对象的使用耦合在一起,职责过重,有违单一职责原则
Java设计模式 —— 单例模式的更多相关文章
- java设计模式单例模式 ----懒汉式与饿汉式的区别
常用的五种单例模式实现方式 ——主要: 1.饿汉式(线程安全,调用率高,但是,不能延迟加载.) 2.懒汉式(线程安全,调用效率不高,可以延时加载.) ——其他: 1.双重检测锁式(由于JVM底层内部模 ...
- Java设计模式の单例模式
-------------------------------------------------- 目录 1.定义 2.常见的集中单例实现 a.饿汉式,线程安全 但效率比较低 b.单例模式的实现:饱 ...
- JAVA设计模式-单例模式(Singleton)线程安全与效率
一,前言 单例模式详细大家都已经非常熟悉了,在文章单例模式的八种写法比较中,对单例模式的概念以及使用场景都做了很不错的说明.请在阅读本文之前,阅读一下这篇文章,因为本文就是按照这篇文章中的八种单例模式 ...
- Java设计模式 - - 单例模式 装饰者模式
Java设计模式 单例模式 装饰者模式 作者 : Stanley 罗昊 [转载请注明出处和署名,谢谢!] 静态代理模式:https://www.cnblogs.com/StanleyBlogs/p/1 ...
- 【设计模式】Java设计模式 - 单例模式
[设计模式]Java设计模式 - 单例模式 不断学习才是王道 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 分享学习心得,欢迎指正,大家一起学习成长! 原创作品,更多关注我CSDN: ...
- Java 设计模式 —— 单例模式
1. 概念: 单例模式是一种常用的软件设计模式.核心结构中只包含一个被称为单例的特殊类.通过单例模式可以保证系统中一个类只有一个实例而且该实例易于外界访问,从而方便对实例个数的控制并节约系统资源.如果 ...
- Java设计模式 - 单例模式 (懒汉方式和饿汉方式)
概念: Java中单例模式是一种常见的设计模式,单例模式的意思就是只有一个实例.单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例.这个类称为单例类. 单例模式的写法有好几种,这 ...
- java设计模式——单例模式(一)
一. 定义与类型 定义:保证一个类仅有一个实例,并提供一个全局访问点 类型:创建型 二. 适用场景 想确保任何情况下都绝对只用一个实例 三. 优缺点 优点: 在内存里只有一个实例,减少了内存开销 可以 ...
- JAVA设计模式--单例模式
单例设计模式 Singleton是一种创建型模式,指某个类采用Singleton模式,则在这个类被创建后,只可能产生一个实例供外部访问,并且提供一个全局的访问点. 核心知识点如下: (1) 将采用单例 ...
- Java设计模式-单例模式(Singleton)
单例对象(Singleton)是一种常用的设计模式.在Java应用中,单例对象能保证在一个JVM中,该对象只有一个实例存在.这样的模式有几个好处: 1.某些类创建比较频繁,对于一些大型的对象,这是一笔 ...
随机推荐
- idea中 .gitignore文件的使用
idea中 .gitignore文件的使用 首先保证当前的所有文件都没有被git追踪 如果被追踪,建议先取消git的版本控制 输入如下指令 find . -name ".git" ...
- maven远程debug
1.修改tomcat服务器配置 打开tomcat/bin/catalina.sh 添加参数 CATALINA_OPTS="-Xdebug -Xrunjdwp:transport=dt_soc ...
- md文件使用说明
md文件简单使用介绍 二级标题 三级标题 斜体文本 粗体文本 粗斜体文本 分隔线 删除号 带下划线 创建脚注格式类似这样 [1]. #include <iostream> using na ...
- 01.数据库基础、JDBC
一.数据库 数据库:用于存储和管理数据的仓库. 数据库的特点 持久化储存数据,数据库就是一个文件系统. 方便储存和管理数据. 使用了统一的方式操作数据库--SQL. 配置 Mysql 服务启动 手动 ...
- 穿透式监管与CTP
https://blog.csdn.net/wowotuo/article/details/90454013 代码示例: https://tashaxing.blog.csdn.net/article ...
- Java学习笔记(二)环境
卸载JDK 1.删除java的安装目录 2.删除JAVA_HOME 3.删除path下关于java的目录 4.java -version 配置环境变量 1.我的电脑-->右键-->属性 ...
- 帝国CMS 登录后跳转ecmsadmin.php 白页 多个解决方案
帝国CMS 6.6 版本, 运维的网站查询除了问题, 把网站数据库和整体网站拷贝到本地服务器, 配置好 e/config/config.php 和 e/class/config.php 两个目录 ...
- 使用IDEA的webservice工具解析生成的客户端调用远程接口
由于这个接口的报文格式比较麻烦,是XML的请求头加上JSON格式的请求体,所以看起来比较复杂,也可以用RPC的方式调用,那样需要将请求头和请求体,响应头和响应体建实体.public JSONObjec ...
- Mac 系统下 xxx.py 在终端运行
1.在文件中添加注释首先在你所要运行的python文件里首行添加一个特殊的注释(我使用的是python 3.7.3) #!/usr/bin/env python3 注意:如果是python3的话,&q ...
- 微信公众号授权登录,整合spring security
公司的业务需求,对接了微信公众号授权,通过微信公众号的接口拿到用户信息进行业务系统的登录,话不多说上代码,我的实现方式是整合了spingSecurity 首先是接口 @PostMapping(&quo ...