java双重检查锁定的实现代码
在Java程序中,有时候可能需要推迟一些高开销的对象初始化操作,并且只有在使用这些对象时才进行初始化 。这称为延迟初始化或懒加载
看一个不安全的延迟初始化:
A线程执行1后,发现对象instance为null,准备对其new,而B线程却先new了,这造成了错误
我们可以利用同步锁,保证正确:
但是对整个方法进行同步开销太大,人们想出了双重检查锁定:
最小范围所用同步锁,利用双重检查看似实现了目的,但这出现了一个问题:当A线程4执行时,线程B的7还未执行完成,而线程A判定instance != null. 线程B的7还未执行完成,为什么会出现这种情况?
看一下new Instance()的底层关键实现:
其实是先执行1分配内存,然后再初始化对象和设置instance.然后这里存在重排,2和3的顺序可能被调换:
所以当B还执行完7时,A在4判定instance对象已经完成初始化了,如果在ctorInstance(memory)之前去调用instance就会出错。
解决办法有两个:
1.将instance对象声明为volatile,它会禁止2,3的重排
2.利用基于类初始化的解决方案 :JVM在类的初始化阶段(即在Class被加载后,且被线程使用之前),会执行类的初始化。在
执行类的初始化期间,JVM会去获取一个锁。这个锁可以同步多个线程对同一个类的初始化
我们会发现基于类初始化的方案的实现代码更简洁。但基于volatile的双重检查锁定的方案有一个额外的优势:除了可以对静态字段实现延迟初始化外,还可以对实例字段实现延迟初始化。字段延迟初始化降低了初始化类或创建实例的开 销,但增加了访问被延迟初始化的字段的开销。在大多数时候,正常的初始化要优于延迟初始化。如果确实需要对实例字段使用线程安全的延迟初始化,请使用上面介绍的基于volatile的延迟初始化的方案;如果确实需要对静态字段使用线程安全的延迟初始化,请使用上面介绍的基于类初始化的方案。
总结
以上所述是小编给大家介绍的java双重检查锁定的实现代码,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对脚本之家网站的支持!
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释
- python抓取文件夹的所有文件
- 深入理解Spring的事务传播行为
- 关于ES6箭头函数中的this问题
- Android Jni的简单使用详解
- Linux中BT5关于修改静态IP和DNS方法
- 在Android中使用Anntation来代替ENUM的方法
- python定向爬取淘宝商品价格
- Android代码混淆的写法总结
- Android使用Handler实现定时器与倒计时器功能
- jQuery幻灯片插件owlcarousel参数说明中文文档
- 详解EventBus 3.x 的快速使用
- Java并发系列之Semaphore源码分析
- vue2.5.2使用http请求获取静态json数据的实例代码
- Vue2.5通过json文件读取数据的方法
- Java并发系列之AbstractQueuedSynchronizer源码分析(条件队列)