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. UI自动化之【maven+selenium环境搭建】

    一.下载maven包 官网: http://maven.apache.org/download.cgi 二.配置maven环境变量  配置完之后验证一下:(若出现以下信息可看到maven的版本号就表示 ...

  2. JS笔记(三):函数与对象

    镇楼图 Pixiv:torino 四.Function类型 Rest语法 一些函数如Math.max可以支持任意数量的参数,JS中对于这样的参数可以简单使用...来实现,使用剩余参数,它支持收集剩余的 ...

  3. 第12组 Beta冲刺 (2/5)

    1.1基本情况 ·队名:美少女战士 ·组长博客:https://www.cnblogs.com/yaningscnblogs/p/14016602.html ·作业博客:https://edu.cnb ...

  4. MPC

    Just for anyone searching for code. I found it here: https://drive.google.com/drive/folders/0BzLEHBD ...

  5. 鸣人的影分身【按照前i个数,最小数是不是0,建立转移方程】

    鸣人的影分身 题意 鸣人最多有n个分身,m的能量.分身的能量可以为0. 问有多少种方案数. 思路 很容易定义状态:f[i] [j]: 前i个分身,共花费能量j的方案数. 状态转移:刚开始想的枚举第i个 ...

  6. 关于Lua中的面向对象实现

    写在前面 最近在琢磨"Lua热重载",在测试中发现我之前对Lua中的面向对象实现有一些理解发生变化,这里记录一下. 本文提到的面向对象实现来自云风. 类实现 <Lua程序设计 ...

  7. 【内存管理】CMA内存分配器(Contiguous Memory Allocator)

    什么是CMA 参考这两篇博文,写得很好: http://www.wowotech.net/memory_management/cma.html https://www.cnblogs.com/Loye ...

  8. HTTP 协议相关

    一. HTTP常见请求头 1. Host (主机和端口号) 2. Connection (连接类型) 3.Upgrade-Insecure-Requests (升级为HTTPS请求) 4. User- ...

  9. java8 stream按某个字段分组,允许分组字段是null

    Map<String, List<Dto>> deviceMap = deviceList.stream().collect(Collectors.groupingBy(Dto ...

  10. 截取屏幕 转为GIF 图片

    近期winform 做的一个截取屏幕的软件给大家!谁要留言给我哦! sss