第 6 章 单例模式与多线程

本章主要内容

如何使单例模式遇到多线程是安全的、正确的。

6.1 立即加载 / “饿汉模式”

  什么是立即加载?立即加载就是使用类的时候已经将对象创建完毕,常见的实现办法就是直接 new 实例化。而立即加载也称为“饿汉模式”。

6.2 延迟加载 / “懒汉模式”

  什么是延迟加载?延迟加载就是在调用 get() 方法时实例才被创建,常见的实现办法就是在 get() 方法中进行 new 实例化。延迟加载也称为“懒汉模式”。

1. 延迟加载 / “懒汉模式” 解析

  延迟加载 / “懒汉模式” 是在调用方法时实例才被创建。

  这种模式如果是在多线程的环境中,就会出现取出多个实例的情况,与单例模式的初衷是相背离的。

2. 延迟加载 / “懒汉模式” 的缺点

  “延迟加载”在多线程的环境中,根本不能实现保持单例的状态。

3. 延迟加载 / “懒汉模式”的解决方案

(1)声明 synchronized 关键字

  既然多个线程可以同时进入 getInstance() 方法,那么只需要对 getInstance() 方法声明 synchronized 关键字即可。

  为 getInstance() 方法加入同步 synchronized 关键字得到相同实例的对象,但此种方法的运行效率非常低下,是同步运行的,下一个线程想要取得对象,则必须等上一个线程释放锁之后,才可以继续执行。

(2)尝试同步代码块

  同步方法是对方法的整体进行持锁,这对运行效率来讲是不利的。

  同步 synchronized 语句块得到相同实例的对象,但此种方法的运行效率也是非常低的,和 synchronized 同步方法一样是同步运行的。

(3)针对某些重要的代码进行单独的同步

  同步代码块可以针对某些重要的代码进行单独的同步,而其他的代码则不需要同步。这样在运行时,效率完全可以得到大幅提升。

  此方法使用同步 synchronized 语句块,只对实例化对象的关键代码进行同步,从语句的结构上来讲,运行的效率的确得到了提升。但如果是遇到多线程的情况下还是无法解决得到同一个实例对象的结果。

(4)使用 DCL 双检查锁机制

  使用 DCL 双检查锁机制来实现多线程环境中的延迟加载单例设计模式。

  DCL 双检查模式就是一方面使用 synchronized 语句款,只对实例化对象的关键代码进行同步,另一方面对单例对象使用 volatile 关键字。

  使用双重检查锁功能,成功地解决了“懒汉模式”遇到多线程的问题。DCL 也是大多数多线程结合单例模式使用的解决方案。

6.3 使用静态内置类实现单例模式

  DCL 可以解决多线程单例模式的非线程安全问题。当然,使用其他的办法也能达到同样的效果。

public class MyObject {
//内部类方式
public static class MyObjectHandler {
private static MyObject myobject = new MyObject();
}
private MyObject() { }
public static MyObjectgetInstance() {
return MyObjectHandler.myObject;
}
}

6.4 序列化与反序列化的单例模式实现

  静态内置类可以达到线程安全问题,但如果遇到序列化对象时,使用默认的方式运行得到的结果还是多例的(单例类 implements Serializable)。

  文件读和写获取的对象实例不同,解决办法就是在反序列中使用 readResolve() 方法。

6.5 使用 static 代码块实现单例模式

  静态代码块中的代码在使用类的时候就已经执行了,所以可以应用静态代码块的这个特性来实现单例设计模式。

static {
instance = new MyObject();
}

6.6 使用 enum 枚举数据类型实现单例模式

  枚举 enum 和静态代码块的特性相似,在使用枚举类时,构造方法会被自动调用,也可以应用其这个特性实现单例设计模式。

public enum MyObject{
...
private MyObject(){
//单例初始化
}
}

6.7 完善使用 enum 枚举实现单例模式

  前面一节将枚举类进行暴露,违反了“职责单一原则”。

public class MyObject{
public enum MyEnumSingletion{
...
private MyEnumSingletion(){
//单例初始化
}
}
}

6.8 本章总结

  本章使用若干案例来阐述单例模式与多线程结合时遇到的情况与解决方法。

Java多线程编程核心技术-第6章-单例模式与多线程-读书笔记的更多相关文章

  1. Java多线程编程核心技术-第4章-Lock的使用-读书笔记

    第 4 章 Lock 的使用 本章主要内容 ReentrantLocal 类的使用. ReentrantReadWriteLock 类的使用. 4.1 使用 ReentrantLock 类 在 Jav ...

  2. Java多线程编程核心技术-第3章-线程间通信-读书笔记

    第 3 章 线程间通信 线程是操作系统中独立的个体,但这些个体如果不经过特殊的处理就不能成为一个整体.线程间的通信就是成为整体的必用方案之一,可以说,使线程间进行通信后,系统之间的交互性会更强大,在大 ...

  3. Java多线程编程核心技术-第1章-Java多线程技能-读书笔记

    第 1 章 Java 多线程技能 本章主要内容 线程的启动 如何使线程暂停 如何使线程停止 线程的优先级 线程安全相关的问题 1.1 进程和多线程的概念及线程的优点 进程是操作系统结构的基础:是一次程 ...

  4. Java编程思想——第17章 容器深入研究 读书笔记(三)

    七.队列 排队,先进先出. 除并发应用外Queue只有两个实现:LinkedList,PriorityQueue.他们的差异在于排序而非性能. 一些常用方法: 继承自Collection的方法: ad ...

  5. Java多线程编程核心技术-第7章-拾遗增补-读书笔记

    第 7 章 拾遗增补 本章主要内容 线程组的使用. 如何切换线程状态. SimpleDataFormat 类与多线程的解决办法. 如何处理线程的异常. 7.1 线程的状态 线程对象在不同的运行时期有不 ...

  6. Java多线程编程核心技术-第5章-定时器 Timer-读书笔记

    第 5 章 定时器 Timer 定时 / 计划功能在移动开发领域使用较多,比如 Android 技术.定时计划任务功能在 Java 中主要使用的就是 Timer 对象,他在内部使用多线程的方式进行处理 ...

  7. Java多线程编程核心技术-第2章-对象及变量的并发访问-读书笔记

    第 2 章 对象及变量的并发访问 本章主要内容 synchronized 对象监视器为 Object 时的使用. synchronized 对象监视器为 Class 时的使用. 非线程安全是如何出现的 ...

  8. java多线程编程核心技术——第三章

    第一节等待/通知机制 1.1不使用等待/通知机制实现线程间的通讯 1.2什么是等待/通知机制 1.3等待/通知机制的实现 1.4方法wait()锁释放与notify()锁不释放 1.5当interru ...

  9. java多线程编程核心技术——第七章补漏拾遗

    本章主要知识点: 1)线程组的使用 2)如何切换线程状态 3)SimpleDateFormat类与多线程的解决方法 4)如何处理线程异常. 这本书基本来到了终点,其实在第四章来说,核心(基础)的线程知 ...

随机推荐

  1. 阿里开源的缓存框架JetCache

    之前一直在用Spring Cache进行接口数据的缓存,主要是Spring Cache在对具体key缓存失效时间的设置不是很方法,还要自己去扩展,无意中发现了阿里的JetCache.大部分的需求都能满 ...

  2. Spring security 知识笔记【内存角色授权】

    一.原有的配置文件中,增加注解@EnableGlobalMethodSecurity(prePostEnabled = true) 二.原有配置文件中,内存新建账号的时候添加角色 package El ...

  3. Hanlp分词插件docker集群安装

    背景:我是用docker-compose的方式装的es集群,正常情况es镜像没有插件,如果在docker里面用命令安装了那么重启以后又没了,所以采用挂载离线安装的方式 版本: es7.2 1下载Han ...

  4. SpringBoot第十九篇:邮件服务

    作者:追梦1819 原文:https://www.cnblogs.com/yanfei1819/p/11118340.html 版权声明:本文为博主原创文章,转载请附上博文链接! 引言   邮件的重要 ...

  5. Python处理数据集-2

    原数据集的数据格式: 每行为:(test_User, test_Item) negativeItem1 negativeItem2 negativeItem3 …… negativeItem99 即每 ...

  6. java识别死亡或者存活的对象

    那些内存需要回收 内存回收是对运行时内存区域的内存回收,其中程序计数器.虚拟机栈.本地方法栈3个区域随线程而生,随线程而灭:栈中的栈帧随着方法的进入和退出而有条不紊的执行着出栈和入栈操作.每一个栈帧中 ...

  7. idea类存在找不到解决办法

    清除idea缓存,

  8. SQLServer查看分区表详细信息

    SQL查看分区内记录个数,常规方法需要知道分区函数然后再显示,网上看到一个一句话显示的方法 ), ps.name ) as partition_scheme, p.partition_number, ...

  9. Scala Types 2

    存在类型 形式: forSome { type ... } 或 forSome { val ... } 主要为了兼容 Java 的通配符 示例 Array[_] // 等价于 Array[T] for ...

  10. 创建Maven项目时,GroupId和Artifact Id该怎么填写呢?

    1.什么是groupid和artifactId? groupid和artifactId被统称为“坐标”是为了保证项目唯一性而提出的,如果你要把你项目弄到maven本地仓库去,你想要找到你的项目就必须根 ...