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. Navicat连接Mysql报错:Client does not support authentication protocol requested by server(转载)

    Navicat连接MySQL Server8.0版本时出现Client does not support authentication protocol requested  by server:解决 ...

  2. 11.30linux学习第十一天

    今天老刘上课,第7章收尾,第8章开了个头. 7.1.3  磁盘阵列+备份盘 RAID 10磁盘阵列中最多允许50%的硬盘设备发生故障,但是存在这样一种极端情况,即同一RAID 1磁盘阵列中的硬盘设备若 ...

  3. api进阶Day2(低级流)文件流的输出流、读取流。向文件中写入文本数据、读取文件中的字符串、用lambda表达式创建文件过滤器。

    文件流:输出流: package io; import java.io.FileNotFoundException; import java.io.FileOutputStream; import j ...

  4. Linux 提示符后面显示全路径

    vi /root/.bashrc # .bashrc # User specific aliases and functions alias rm='rm -i' alias cp='cp -i' a ...

  5. TP5.1模板循环标签

    第一种volist name=assign中的变量名 id=数组中的key offset=开始循环的位置 length=步长 {volist name='list' id='vo' offset='0 ...

  6. 截取屏幕 转为GIF 图片

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

  7. redis学习(一)常用命令

    clear 清屏 info 获取当前redis的版本等信息 keys pattern 查找所有符合给定模式( pattern)的 key keys * 查找所有的key type key 查看某个ke ...

  8. k8s 关于pull image failed 问题

    问题描述: Failed to pull image "nginx": rpc error: code = Unknown desc = failed to pul 解决办法: 1 ...

  9. Windows 下TCP长连接保持连接状态TCP keepalive设置

    TCP长连接建立完成后,我们通常需要检测网络的连接状态,以反馈给客户做响应的处理.通过设置TCP keepalive的属性,打开socket的keepalive属性,并设置发送底层心跳包的时间间隔.T ...

  10. Linux下apache日志(按日期存放)分析与状态查看方法

    转载网址: https://blog.csdn.net/weixin_42272246/article/details/125602258