java锁以及双重检查

双检锁/双重校验锁 双层对空判断困扰了很久。实例

public class Singleton {private volatile static Singleton singleton;//私有构造函数避免调用private Singleton (){}public static Singleton getSingleton() {// 先判断对象是否创建过if (singleton == null) {//类对象加锁synchronized (Singleton.class) {if (singleton == null) {singleton = new Singleton();//字节码层//JIT CPU 可能对如下指令进行重排序// 1.分配空间// 2.初始化// 3.引用赋值如果指令重排后指令如下://1.分配空间//3.引用赋值 如果在当前指令执行完后,有其他线程获取到实例,将拿到未初始化的实例;//2. 初始化}}}return singleton;}
}


解释:当A与B同时调用getSingleton时,判断第一个if都为空,这时A拿到锁,进行第二层if判断,条件成立new了一个对象;

B在外层等待,A创建完成,释放锁,B拿到锁,进行第二层if判断,条件不成立,结束释放锁。C调用getSingleton时第一层判断不成立,直接拿到singleton对象返回,避免进入锁,减少性能开销。

进一步理解:其中两次判空,第一次判空是,减少多线程情况下,进入同步代码块的次数,第二次判空,是防止多线程(A,B两种线程的情况下A,B 同时调用了 getSingleton 方法,都同时,进入到第一层if(singleton==null){} 内,竞态条件下,如果A 拿到了对象锁,进入到同步代码块,B阻塞等待,等待创建了实例对象,释放了锁后,B进入同步代码块,但是此时第二层if(singleton==null){} 判断,singleton 不为空,直接返回singleton);

总结:其中有两次判断是否为空的语句,第一次是为了提高效率,避免每次都要执行同步代码块,第二次判空,是为了避免多线程带来的不安全,当两个线程同时对第一个判断为空时,均会先后进入同步代码块,此时,若没有第二个判空条件,则会引来创建多个实例。

volite 关键字:

采用volatile关键字修饰很有必要

这句代码事实上是分三步:

singleton = new Singleton();

  1. 为singleton 分配内存空间
  2. 初始化singleton 
  3. 将singleton 指向分配的内存地址
    但是,jvm会有指令重排的特性,执行顺序有可能改变,不是按照123的顺序,可能是132,这样就会导致一个线程获得没有初始化的实例
    如:t1执行了13,此时t2调用了getInstance()后发现singleton 已经不为空了,因此返回singleton ,但是这时singleton 还没有被初始化
    volatile就可以禁止jvm的指令重排,保证在多线程环境下也能正常运行

想进一步了解:推荐

深入理解Java并发之synchronized实现原理

https://blog.csdn.net/javazejian/article/details/72828483

 

参考:

https://blog.csdn.net/jlnu0812407/article/details/102836187

https://blog.csdn.net/faye_1008/article/details/90296173

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/423242.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【C++深度剖析教程24】C++中不同的继承方式

过完年了,今天开始写博客记录学习的过程。继续C的学习,今天我们来看C中不同的继承方式。 一、初探继承的方式 从上图看,是否可以将继承中的public换成protected或者private。如果可以,它们与public继承的区别是什么? …

前端学习(97):psd切图流程

编辑----首选项---增效工具----启用生成器 文件---生成----图像资源 选择图层修改命名为png,则再默认路径下生成。 默认为半透明

JavaScript:constructor属性

constructor属性始终指向创建当前对象的构造函数。比如下面例子: //等价于 var foo new Array(1, 56, 34, 12); vararr [1, 56, 34, 12]; console.log(arr.constructor Array); //true //等价于 var foo new Function(); varFoo function() { }; console.log(Foo.…

IT职业就业-学长有话说(二)

对应刚毕业就人传统行业的,同学需要谨慎,因为传统公司的技术,基本上是一年或者两年,基本上就会遇到瓶颈,看上去,似乎已经掌握了,许多技术和框架,但是,相对于技术的深度确…

【C++深度剖析教程25】继承中的构造与析构

今天来学习C中继承的构造与析构,有兴趣一起学习的加qq:1126137994 1、问题 如何初始化父类成员?父类构造函数与子类构造函数有什么关系? 子类对象是如何构造的? 子类中可以定义构造函数子类构造函数必须对继承而来…

springboot创建子模块时遇到子模块覆盖父模块问题解决

1.最近更新git dev 开发分支需要添加一个子模块,创建过程中遇到 创建springboot 模块时,子模块覆盖了父模块,搞了半天,最后查询到解决办法如下: 1 多模块项目创建 因为本系列的下一篇是《Spring Boot集成Dubbo》&…

[MySQL] - 返回影响行数

(在MySQL 5.1.36上测试) found_rows() : selectrow_count() : insert update delete 注:需要配合相应的操作一起使用,否则返回的值只是1和-1(都是不正确的值) 示例: dropdatabaseifexistsmytest;createdatabasemytest;usemytest;droptableifexistsMyTestTable;createtableMyTe…

【C++深度剖析教程26】父子间的冲突

今天继续来学习C,父类与子类之间的冲突。加qq1126137994共同学习交流。 1、问题 子类中是否可以定义父类中的同名成员,如果可以,如何区分?如果不可以那又为什么? 2、问题的延伸 子类可以定义父类中的同名成员子类中…

java编程思想学习(1):抽象

今天开始java编程思想的导读,希望自己能更好的理解程序。全书1461页,恐怖,尽量一天看十页八。 OOP 面向对象程序设计 记录一点东西八 第一个小课程,抽象过程 看看概念 1万物皆为对象 2程序是对象的集合,他们通过…

嵌入式Linux系统之I.MX6触摸屏驱动程序TSC2007.C的分析、移植与校准

学习交流加 个人qq: 1126137994个人微信: liu1126137994学习交流资源分享qq群: 962535112 今天来记录一下I.MX6开发板移植触摸屏驱动程序的过程分析。在移植驱动程序之前,为了学习,先去分析一下触摸屏驱动程序的框架。…

docker 安装及打springboot jar打镜像

1.首先是安装 centos7 2.安装docker 建议参考 https://www.jianshu.com/p/ef14131fe900 2.1docker 一些常用的命令 docker ps 查看容器 docker run 创建容器 docker rmi imageId 删除容器 docker images 列出索引镜像 各个选项说明: REPOSITORY:表示镜像的…

java编程思想学习(2):对象

按照课本上的案例 我们也来创建一个小灯泡的案例 package geyao02;public class Light {public void on() {System.out.println("我可以发光");}public void off() {System.out.println("我可以关闭");}}运行结果 package geyao02;public class TestLig…

【C++深度剖析教程27】多态的概念与意义

今天来学习一个新的概念,多态!!!多态在C编程中具有重要的地位与意义,是面向对象的一个重要思想! 加qq1126137994一起学习更多技术~ 1、问题引入 父类与子类之间具有赋值兼容性; *子类对象可以当做父类对…

母亲的革命

做了几个星期思想工作,经过一天的舟车之劳,终于把母亲接到了县城住下来。 按理说,老早就想接母亲出来享享清福,顺便带带金果。可母亲一来放心不下家里的土地,二来怕婆媳之间不好相处,所以都推绝了。 母亲今…

springboot+hbase 集成

项目中使用 phoenix 使用SQL 方式来操作Hbase 数据库,但是遇到一个是,SQL在Dbeaver 中查询速度还可以,但是使用phoenixibatis 后返回结果集数据量20w ,速度特别慢,先是考虑用redis方式缓存,但是内存有限&am…

java编程思想学习(3):Java中的private、protected、public和default的区别

(1)对于public修饰符,它具有最大的访问权限,可以访问任何一个在CLASSPATH下的类、接口、异常等。它往往用于对外的情况,也就是对象或类对外的一种接口的形式。 (2)对于protected修饰符&#xf…

【Makefile由浅入深完全学习记录2】初识 makefile 的结构

继续学习makefile,希望尽快掌握makefile好在面试中更好的掌握方向!加qq1126137994一起学习更多技术!! 1、回顾 makefile的意义: makefile用于定义源文件之间的依赖关系makefile说明如何编译各个源文件并生成可执行文…

只进ResultSet 不支持请求的操作

使用jdbc操作时 抛出异常 只进ResultSet 不支持请求的操作 那肯定是 在使用resultset的游标操作时调用 resultSet.first () resultSet.last等等方法 你要做的就是设置游标支持滚动操作 如果是statement Statement stmtcon.createStatement(ResultSet.TYPE_SCROLL_…

【Makefile由浅入深完全学习记录3】伪目标的引入

只有不停的努力,才能看起来很轻松!!!加qq1126137994共同学习探讨更多技术!!! 1、思考: makefile中的目标究竟是什么? 在默认的情况下: make认为目标对应一…

InnoDB的auto_increment指定值被重置问题

有时候新建的表需要对自增列指定初始值,但是有时候会出现明明指定过的初始值却被重置的现象。下面以一个小实验来说明这个问题:MySQL version:5.1.42 OS:redhat5.3无废话,建张表先:代码CREATETABLEsbtest1 ( id int(10) unsigned…