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设计模式 —— 单例模式的更多相关文章

  1. java设计模式单例模式 ----懒汉式与饿汉式的区别

    常用的五种单例模式实现方式 ——主要: 1.饿汉式(线程安全,调用率高,但是,不能延迟加载.) 2.懒汉式(线程安全,调用效率不高,可以延时加载.) ——其他: 1.双重检测锁式(由于JVM底层内部模 ...

  2. Java设计模式の单例模式

    -------------------------------------------------- 目录 1.定义 2.常见的集中单例实现 a.饿汉式,线程安全 但效率比较低 b.单例模式的实现:饱 ...

  3. JAVA设计模式-单例模式(Singleton)线程安全与效率

    一,前言 单例模式详细大家都已经非常熟悉了,在文章单例模式的八种写法比较中,对单例模式的概念以及使用场景都做了很不错的说明.请在阅读本文之前,阅读一下这篇文章,因为本文就是按照这篇文章中的八种单例模式 ...

  4. Java设计模式 - - 单例模式 装饰者模式

    Java设计模式 单例模式 装饰者模式 作者 : Stanley 罗昊 [转载请注明出处和署名,谢谢!] 静态代理模式:https://www.cnblogs.com/StanleyBlogs/p/1 ...

  5. 【设计模式】Java设计模式 - 单例模式

    [设计模式]Java设计模式 - 单例模式 不断学习才是王道 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 分享学习心得,欢迎指正,大家一起学习成长! 原创作品,更多关注我CSDN: ...

  6. Java 设计模式 —— 单例模式

    1. 概念: 单例模式是一种常用的软件设计模式.核心结构中只包含一个被称为单例的特殊类.通过单例模式可以保证系统中一个类只有一个实例而且该实例易于外界访问,从而方便对实例个数的控制并节约系统资源.如果 ...

  7. Java设计模式 - 单例模式 (懒汉方式和饿汉方式)

    概念: Java中单例模式是一种常见的设计模式,单例模式的意思就是只有一个实例.单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例.这个类称为单例类. 单例模式的写法有好几种,这 ...

  8. java设计模式——单例模式(一)

    一. 定义与类型 定义:保证一个类仅有一个实例,并提供一个全局访问点 类型:创建型 二. 适用场景 想确保任何情况下都绝对只用一个实例 三. 优缺点 优点: 在内存里只有一个实例,减少了内存开销 可以 ...

  9. JAVA设计模式--单例模式

    单例设计模式 Singleton是一种创建型模式,指某个类采用Singleton模式,则在这个类被创建后,只可能产生一个实例供外部访问,并且提供一个全局的访问点. 核心知识点如下: (1) 将采用单例 ...

  10. Java设计模式-单例模式(Singleton)

    单例对象(Singleton)是一种常用的设计模式.在Java应用中,单例对象能保证在一个JVM中,该对象只有一个实例存在.这样的模式有几个好处: 1.某些类创建比较频繁,对于一些大型的对象,这是一笔 ...

随机推荐

  1. 2月26日Android学习

    今天下载了AndroidStudio,但是不知道为什么java文件一直显示file outside of source root,可能是部分文件没有安装的原因,等文件全部下载完之后再看看.

  2. oracle从1到10生成顺序号

    oracle从1到10生成顺序号,脚本案例如下: select 'ABAB'|| lpad(level,5,0) as serial_no from dual connect by level< ...

  3. Docker安装(win10)

    Docker安装 windows 首先开启Hyper-V Hyper-V 是微软开发的虚拟机,类似于 VMWare 或 VirtualBox,仅适用于 Windows 10.这是 Docker Des ...

  4. Docker部署Nginx报错 WARNING: IPv4 forwarding is disabled. Networking will not work.

    Docker 部署 Nginx 报错 WARNING: IPv4 forwarding is disabled. Networking will not work. [root@localhost ~ ...

  5. (四).JavaScript的循环结构

    2.2 循环嵌套 ①.语法 // 嵌套循环:循环内部包裹其他的循环 // 外侧循环执行一次,内部循环执行一轮 // 实例 for (var i = 0; i < 5; i++) { for (v ...

  6. docker 镜像rabbitmq安装

    docker 镜像rabbitmq安装 1.拉取镜像 带有"mangement"的版本(包含web管理页面): docker pull rabbitmq:3.7.7-managem ...

  7. docker&docker-compose安装

    一.docker安装 1.通过 uname -r 命令查看当前的内核版本,Docker 要求 CentOS 系统的内核版本高于 3.10 uname -r 2.查看系统是否安装过docker yum ...

  8. supervisor 使用中遇到的问题

    supervisor 配置完毕,使用supervisorctl reload 和supervisorctl update 启动时候报错 解决方法使用下面命令启动 /usr/bin/python2 /u ...

  9. 【StoneDB 模块介绍】服务器模块

    [StoneDB 模块介绍]服务器模块 一.介绍 客户端程序和服务器程序本质上都是计算机上的一个进程,客户端进程向服务器进程发送请求的过程本质上是一种进程间通信的过程,StoneDB 数据库服务程序作 ...

  10. 9. 实现包括前端后台的预约洗狗功能 - 使用Power Automate发送预约邮件 - 使用Power Automate发送带选择按钮(option)的邮件

    ​ 除了发送普通的电子邮件外,我们还可以选择发送带选项的电子邮件来得到客户的反馈,下面我们就一起来创建带有选择功能的电子邮件吧. 1. 打开我们的Power Portal,在左侧导航栏选择流,点击左上 ...