设计模式课程 设计模式精讲 8-3 单例设计模式-DoubleCheck双重检查实战及原理解析
1 课程讲解
1.1 为何要使用双重检查
1.2 双重检查的缺点
1.3 指令重排序讲解
1.4 指令重排序比喻(自己理解)
1.5 如何解决指令重排序问题
2 代码演练
2.1 代码演练1(双重检查 解决对象锁和类锁的问题)
2.2 代码演练2(volatile 应用:解决重排序问题)
1 课程讲解
1.1 为何要使用双重检查
在上节课的时候,多线程的时候,由于一个线程被锁,其他的线程无法访问该类,被堵塞。性能大大降低,双重检查主要是应用解决此类问题。
双重检查可以使更多的线程堵塞在方法中,而不是在类之外,这样的话,当锁被释放的时候,能够更快的执行,可以大大的提高效率。
1.2 双重检查的缺点
java在执行过程中,可能会出现指令重排序问题,(后边语句参考本节2.1代码演练2)导致a线程(先来的)的对象已经赋值,但是还没有初始化完成,这时线程b(后到的)经过判断,也开始访问对象(因为现在对象不为空),导致线程b访问的是线程a还未初始化完成的对象。由于对象并没有被完整的初始化上,系统会报异常。
1.3 指令重排序讲解
初始化的时候,实际进行了三个步骤:
a 给该对象分配内存
b 初始化该对象
c 设置该对象指向给该对象分配的内存
一般情况下,按照abc的顺序执行,但是也会有一定几率bc 颠倒。
这在单线程中执行的时候并没有问题,而且能够提高运行的效率。
1.4 指令重排序比喻(自己理解)
可以比喻如下:
人们来吃饭,
a 首先食堂拿出一份饭,
b 确定这个人是谁,
c 这个人拿走这份饭
1.5 如何解决指令重排序问题
两种方法:
a 使重排序不再发生,每个执行的进程都按照初始化的正常步骤进行 参见 本节代码2.2
b 不允许后来的线程 看到 先来的线程进行的重排序问题
2 代码演练
2.1 代码演练1(解决对象锁和类锁的问题)
测试类:
package com.geely.design.pattern.creational.singleton; public class Test { /*public static void main(String [] args){
//这样写异常,因为构造方法私有
// LazySingleton lazySingleton = new LazySingleton();
LazySingleton lazySingleton = LazySingleton.getInstance();
System.out.println(lazySingleton);
}*/ public static void main(String [] args){
Thread thread1 = new Thread(new T());
Thread thread2 = new Thread(new T());
thread1.start();
thread2.start();
System.out.println("结束了!!!");
}
}
线程类:
package com.geely.design.pattern.creational.singleton; /**
* 注:该类为线程类,调用LazySingleton
*/
public class T implements Runnable{ /*@Override
public void run() {
LazySingleton lazySingleton = LazySingleton.getInstance();
System.out.println(Thread.currentThread().getName()+"==="+lazySingleton); }*/ @Override
public void run() {
LazyDoubleCheckSingleton lazyDoubleCheckSingleton = LazyDoubleCheckSingleton.getInstance();
System.out.println(Thread.currentThread().getName()+"==="+lazyDoubleCheckSingleton);
}
}
实体类:
package com.geely.design.pattern.creational.singleton; public class LazyDoubleCheckSingleton {
/*
属性私有,其他外部类,无法调用该属性,安全
*/
private static LazyDoubleCheckSingleton lazyDoubleCheckSingleton = null; /**
* 构造方法私有,其他类无法实例化该类
*/
private LazyDoubleCheckSingleton(){
} /**
* 换一种写法,
*
* @return
*/
public static LazyDoubleCheckSingleton getInstance(){
if(lazyDoubleCheckSingleton == null){
synchronized (LazyDoubleCheckSingleton.class){
if(lazyDoubleCheckSingleton == null){
lazyDoubleCheckSingleton = new LazyDoubleCheckSingleton();
}
}
}
return lazyDoubleCheckSingleton;
} }
打印结果:
"C:\Program Files\Java\jdk1.7.0_79\bin\java.exe" -agentlib:jdwp=transport=dt_socket,address=127.0.0.1:9096,suspend=y,server=n -javaagent:C:\Users\weijingli\.IdeaIC2018.1\system\captureAgent\debugger-agent.jar=file:/C:/Users/weijingli/AppData/Local/Temp/capture.props -Dfile.encoding=UTF-8 -classpath "C:\Program Files\Java\jdk1.7.0_79\jre\lib\charsets.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\deploy.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\ext\access-bridge-64.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\ext\dnsns.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\ext\jaccess.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\ext\localedata.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\ext\sunec.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\ext\sunjce_provider.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\ext\sunmscapi.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\ext\zipfs.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\javaws.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\jce.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\jfr.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\jfxrt.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\jsse.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\management-agent.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\plugin.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\resources.jar;C:\Program Files\Java\jdk1.7.0_79\jre\lib\rt.jar;F:\xiangmu3\Xin\Idea\design_pattern\target\classes;D:\java\devolopKit\idea\anZh\IntelliJ IDEA Community Edition 2018.1.4\lib\idea_rt.jar" com.geely.design.pattern.creational.singleton.Test
Connected to the target VM, address: '127.0.0.1:9096', transport: 'socket'
结束了!!!
Thread-0===com.geely.design.pattern.creational.singleton.LazyDoubleCheckSingleton@8fea539
Disconnected from the target VM, address: '127.0.0.1:9096', transport: 'socket'
Thread-1===com.geely.design.pattern.creational.singleton.LazyDoubleCheckSingleton@8fea539 Process finished with exit code 0
2.2 代码演练2(volatile 应用:解决重排序问题)
java语言规范中规定:所有线程执行java程序时,必须要遵守intra-thread semantics ,
intra-thread semantics 保证重排序不会改变单线程内的程序执行结果。
换句话说,intra-thread semantics 允许那些在单线程内,不会改变单线程程序执行结果的重排序。
package com.geely.design.pattern.creational.singleton; public class LazyDoubleCheckSingleton {
/*
1 volatile关键字的作用
将当前处理器缓存行的数据写回到内存,该操作会使在其他cpu内存中缓存了该内存地址的数据无效。它们又从共享内存同步数据。 如此操作保存内存的可见性。j
2
*/
private volatile static LazyDoubleCheckSingleton lazyDoubleCheckSingleton = null; /**
* 构造方法私有,其他类无法实例化该类
*/
private LazyDoubleCheckSingleton(){
} /**
* 换一种写法,
*
* @return
*/
public static LazyDoubleCheckSingleton getInstance(){
if(lazyDoubleCheckSingleton == null){
synchronized (LazyDoubleCheckSingleton.class){
if(lazyDoubleCheckSingleton == null){
lazyDoubleCheckSingleton = new LazyDoubleCheckSingleton();
}
}
}
return lazyDoubleCheckSingleton;
} }
设计模式课程 设计模式精讲 8-3 单例设计模式-DoubleCheck双重检查实战及原理解析的更多相关文章
- JAVA_SE基础——38.单例设计模式
本文继续介绍23种设计模式系列之单例模式. 我们在javaSE的基础学习中,会讲到:单例设计模式.模板设计模式.装饰者设计模式.观察者设计模式.工厂设计模式 我以后随着水平的提高,我会专门开个分类写设 ...
- 单例设计模式(Singleton)
一.单例设计模式介绍 所谓类的单例设计模式,就是采取一定的方法保证在整个的软件系统中,对某个类只能存在一个对象实例, 并且该类只提供一个取得其对象实例的方法(静态方法) 例如:Hibernate的Se ...
- Java设计模式—单例设计模式(Singleton Pattern)全然解析
转载请注明出处:http://blog.csdn.net/dmk877/article/details/50311791 相信大家都知道设计模式,听的最多的也应该是单例设计模式,这种模式也是在开发中用 ...
- iOS开发之单例设计模式(完整正确版本)
单例的意思从字面上就可以略知一二,所谓单例就是确保在程序运行过程中只创建一个对象实例.可以用于需要被多次广泛或者说多次使用的资源中,比如我们常见的网络请求类.工具类以及其它管理类等.比如我iOS开发中 ...
- JAVA之旅(十四)——静态同步函数的锁是class对象,多线程的单例设计模式,死锁,线程中的通讯以及通讯所带来的安全隐患,等待唤醒机制
JAVA之旅(十四)--静态同步函数的锁是class对象,多线程的单例设计模式,死锁,线程中的通讯以及通讯所带来的安全隐患,等待唤醒机制 JAVA之旅,一路有你,加油! 一.静态同步函数的锁是clas ...
- JAVA之旅(六)——单例设计模式,继承extends,聚集关系,子父类变量关系,super,覆盖
JAVA之旅(六)--单例设计模式,继承extends,聚集关系,子父类变量关系,super,覆盖 java也越来越深入了,大家加油吧!咱们一步步来 一.单例设计模式 什么是设计模式? JAVA当中有 ...
- 【java】设计模式-单例设计模式
单例设计模式:解决一个类在内存中是存在一个对象的问题.当需要该事物的对象在内存中唯一时,将以下三步添加即可. 思想:想要保证对象唯一1.为了避免其他程序过多的建立该类对象,先禁止其他程序建立该类对象2 ...
- 【iOS 单例设计模式】底层解析与运用
[iOS 单例设计模式]底层解析与运用 一.单例设计名词解释: (官方解释)单例模式确保一个类只有一个实例,自行提供这个实例并向整个系统提供这个实例.(形象比喻)程序 — 公司 单例实例 - 管理 ...
- 单例设计模式全局缓存accessToken
使用微信JS-SDK开发的小伙伴们,看文档经常会看到这样一句话:(下面是微信开发文档的一部分原话截图) 这句话就是:开发者必须在自己的服务全局缓存access_token,jsapi_ticket 下 ...
随机推荐
- cvc-complex-type.2.4.c: The matching wildcard is strict, but no declaration can be found for element 'context:component-scan'.
关于以上错误,观察是否缺少了某一项,还要注意书写的顺序 建议这一类配置文件直接复制就好了,不要自己写 <?xml version="1.0" encoding="U ...
- Java面向对象编程 -6.2
数组的引用传递 通过数组的基本定义可以发现,在数组使用的过程中依然需要使用new进行内存空间的开辟,同理,那么也一定存在有内存的关系匹配问题. 但是数组本身毕竟属于引用数据类型,那么既然是引用数据类型 ...
- mysql之mysql的安装
此次MySQL安装的版本为:MySQL8.0 系统为:centos6.9 64位 一.利用yum仓库安装 wget https://repo.mysql.com//mysql80-community- ...
- IDEA自动部署WEB工程至远程服务器(学习笔记)
一.部署Web工程的几种方式 ①本地打war,上传至远程服务器tomcat容器即可 优点:简单粗暴 缺点:浪费时间 ②IDEA自动部署至远程服务器 优点:节省大量时间 缺点:配置稍多(第一次) 二.I ...
- 第八届极客大挑战 Web-故道白云&Clound的错误
web-故道白云 题目: 解题思路: 0x01 首先看到题目说html里有秘密,就看了下源代码如图, 重点在红圈那里,表示输入的变量是id,当然上一行的method=“get”同时说明是get方式获取 ...
- java 编译java文件 以及生成可执行jar
1.新建java project: 2.src下新建包以及class文件: 3.打包: 5.选取目标mainclass 很关键决定jar是否可执行: 7.build jar : 8:artifact ...
- jquery--获取多选框的值、获取下拉多选框的值
获取多选框的值 var packageCodeList=new Array(); $('#server_id:checked').each(function(){ packageCodeList.pu ...
- 【代码学习】PYTHON 函数
一.定义函数 def 函数名(): 代码 二.函数调用 #定义函数 def printme(str): print str return #调用函数 printme("SQYY1" ...
- 吴裕雄 python 神经网络——TensorFlow训练神经网络:花瓣识别
import os import glob import os.path import numpy as np import tensorflow as tf from tensorflow.pyth ...
- spring boot 整合mapreduce运行的ClassNotFoundException
问题 一个wordcount运行总是报错 java.lang.RuntimeException: java.lang.ClassNotFoundException: Class com.hadoop. ...