Java高并发--安全发布对象
Java高并发--安全发布对象
主要是学习慕课网实战视频《Java并发编程入门与高并发面试》的笔记
发布对像:使一个对象能够被当前范围之外的对象使用。
对象逸出:一种错误的发布。当一个对象还没有构造完成时,就使他被其他线程所见。
如何安全的发布对象?
- 在静态初始化函数中初始化一个对象引用
- 将对象的引用保存到volatile类型域或者AtomicReference中
- 将对象的引用保存到某个正确构造对象的final类型域中
- 将对象的引用保存到一个由锁保护的域中
单例模式
拿单例模式来说
@ThreadSafe
public class SingletonImp {
// 饿汉模式
private static SingletonImp singletonImp = new SingletonImp();
// 私有化(private)该类的构造函数
private SingletonImp() {
}
public static SingletonImp getInstance() {
return singletonImp;
}
public static void main(String[] args) {
System.out.println(SingletonImp.getInstance());
}
}
上面是“饿汉模式”的实现。这个实现是线程安全的,但是不能延迟加载。也就是说这个单例在类加载时就已经初始化,而不是在需要用到他的地方再初始化的,这在一定程度上造成了资源的浪费。
单例还经常有一种叫“懒汉模式”,也就是在需要用到的时候才初始化。
@NotThreadSafe
public class SingletonImp2 {
private static SingletonImp2 singletonImp2;
private SingletonImp2() {}
// 懒汉模式
public static SingletonImp2 getInstance() {
if (singletonImp2 == null) {
singletonImp2 = new SingletonImp2();
}
return singletonImp2;
}
public static void main(String[] args) {
System.out.println(SingletonImp2.getInstance());
}
}
这个程序实现了延迟加载,但是在多线程下,可能多个线程同时执行到if (singletonImp2 == null) {}导致同时创建了多个实例对象,这是线程不安全的实现。所以就有了下面的实现
@ThreadSafe
public class SingletonImp3 {
private static SingletonImp3 singletonImp3;
private SingletonImp3() {}
// 懒汉模式
public static SingletonImp3 getInstance() {
// 同步锁的加入
synchronized (SingletonImp3.class) {
if (singletonImp3 == null) {
singletonImp3 = new SingletonImp3();
}
}
return singletonImp3;
}
}
这个实现加了同步锁,因此是线程安全的,但是在实例被创建出来后每次getInstance都会加上同步锁(这是完全没有必要的,因为if条件中的语句不会被执行到了 ),严重拖慢了程序的性能。
完全可以在第一次创建实例时才上锁,其余时候不需要,基于此思想的实现一般称为“双重检测”,如下
@NotThreadSafe
public class SingletonImp4 {
private static SingletonImp4 singletonImp4;
private SingletonImp4() {}
public static SingletonImp4 getInstance() {
// 第一次创建时才加锁
if (singletonImp4 == null) {
synchronized (SingletonImp4.class) {
if (singletonImp4 == null) {
singletonImp4 = new SingletonImp4();
}
}
}
return singletonImp4;
}
}
注意,这个实现是线程不安全的。究其原因,singletonImp4 = new SingletonImp4();这句其实是分三步完成的。
分配对象的内存空间
初始化对象
设置singleton变量指向刚刚分配的内存
但是由于指令重排原因,第2步和第3步可能会被交换(这个顺序在单线程下没有影响)。我们再看这个程序,注意注释部分。假设线程B执行到第一个if处,同时线程A执行到第二个if处,且由于指令重排的原因,只完成了上面三步的第1和第3步,也就是说此时singletonImp4已经被分配了内存,有值了在线程B判断时,就会跳过if直接返回singletonImp4。但是singletonImp4还没有初始化!线程B取得了还未初始化的对象,一旦使用就会出现问题。
@NotThreadSafe
public class SingletonImp4 {
private static SingletonImp4 singletonImp4;
private SingletonImp4() {}
public static SingletonImp4 getInstance() {
if (singletonImp4 == null) { // 线程B
synchronized (SingletonImp4.class) {
if (singletonImp4 == null) { // 线程A
singletonImp4 = new SingletonImp4(); // 线程A
}
}
}
return singletonImp4;
}
}
考虑到volatile关键字可以禁止指令重排,声明singletonImp4时,添加volatile就可以解决这个问题了。
private static volatile SingletonImp4 singletonImp4;
这样volatile + double check才是线程安全的实现。
Java高并发--安全发布对象的更多相关文章
- java高并发系列 - 第13天:JUC中的Condition对象
本文目标: synchronized中实现线程等待和唤醒 Condition简介及常用方法介绍及相关示例 使用Condition实现生产者消费者 使用Condition实现同步阻塞队列 Object对 ...
- Java高并发--线程安全策略
Java高并发--线程安全策略 主要是学习慕课网实战视频<Java并发编程入门与高并发面试>的笔记 不可变对象 发布不可变对象可保证线程安全. 实现不可变对象有哪些要注意的地方?比如JDK ...
- Java高并发综合
这篇文章是研一刚入学时写的,今天整理草稿时才被我挖出来.当时混混沌沌的面试,记下来了一些并发的面试问题,很多还没有回答.到现在也学习了不少并发的知识,回过头来看这些问题和当时整理的答案,漏洞百出又十分 ...
- [ 高并发]Java高并发编程系列第二篇--线程同步
高并发,听起来高大上的一个词汇,在身处于互联网潮的社会大趋势下,高并发赋予了更多的传奇色彩.首先,我们可以看到很多招聘中,会提到有高并发项目者优先.高并发,意味着,你的前雇主,有很大的业务层面的需求, ...
- 【实战Java高并发程序设计 7】让线程之间互相帮助--SynchronousQueue的实现
[实战Java高并发程序设计 1]Java中的指针:Unsafe类 [实战Java高并发程序设计 2]无锁的对象引用:AtomicReference [实战Java高并发程序设计 3]带有时间戳的对象 ...
- 【实战Java高并发程序设计 5】让普通变量也享受原子操作
[实战Java高并发程序设计 1]Java中的指针:Unsafe类 [实战Java高并发程序设计 2]无锁的对象引用:AtomicReference [实战Java高并发程序设计 3]带有时间戳的对象 ...
- 【实战Java高并发程序设计 4】数组也能无锁:AtomicIntegerArray
除了提供基本数据类型外,JDK还为我们准备了数组等复合结构.当前可用的原子数组有:AtomicIntegerArray.AtomicLongArray和AtomicReferenceArray,分别表 ...
- 【实战Java高并发程序设计 3】带有时间戳的对象引用:AtomicStampedReference
[实战Java高并发程序设计 1]Java中的指针:Unsafe类 [实战Java高并发程序设计 2]无锁的对象引用:AtomicReference AtomicReference无法解决上述问题的根 ...
- 【实战Java高并发程序设计 1】Java中的指针:Unsafe类
是<实战Java高并发程序设计>第4章的几点. 如果你对技术有着不折不挠的追求,应该还会特别在意incrementAndGet() 方法中compareAndSet()的实现.现在,就让我 ...
随机推荐
- @ResponseBody 返回乱码 的解决办法
1:最快的 最简单的办法是 在Ajax请求脸面指定头信息Accept属性,StringHttpMessageConverter默认iso-8859-1编码,但是会根据请求头信息指定的编码格式来转换 ...
- 了解Serverless架构
1 概述 Serverless中文译为“无服务”是一种新兴起的架构模式,公司ESB产品引入Rest微服务服务机制过程,笔者刚好参与其中,其中Serverless作为一个新起的概念,跟微服务架构相关,为 ...
- 吴恩达机器学习笔记8-多变量线性回归(Linear Regression with Multiple Variables)--多维特征
我们探讨了单变量/特征的回归模型,现在我们对房价模型增加更多的特征,例如房间数楼层等,构成一个含有多个变量的模型,模型中的特征为(
- PHP中的自动加载
自动加载? 或许你已经对自动加载有所了解.简单描述一下:自动加载就是我们在new一个class的时候,不需要手动去写require来导入这个class.php文件,程序自动帮我们加载导入进来.这是 ...
- 毕业不到一年,绩效打了个D!
周末了,和大家来聊聊程序员工作态度的问题. 说说栈长的事迹吧,这是好多年前的事了,那时候,栈长才毕业不到一年,那次绩效打了个D!事后,我很气愤啊,我那时还在博客上写文章怒骂了部门经理,现在想起来,真是 ...
- java基础-2
java基础-2 面向对象 定义 面向对象是一种思维方式,相对于面向过程面向过程注重流程中的每一步,清楚流程中的每一个细节面向对象注重的是对象,有了对象就有对象的一届自己动手做--面向过程,找其 ...
- remote: Incorrect username or password ( access token )
解决问题 进入控制面板 用户账号,选择管理您的凭据 修改凭据 修改完成后,保存即可
- Jexus~mono中使用StackExchange.redis的问题
在windows平台的vs里,添加包包时,可以选择StackExchange.redis,而如果你的应该程序需要部署到linux的mono环境上,使用这个StackExchange.redis是不行的 ...
- vue 关于图片路径的问题
在vue 中,当我们想加载assets中的图片,本人按照多年的开发经验会这样写,那是没问题的 <img src="../assets.1.jpg"/> 如果我要用v-b ...
- Docker学习之3——容器
容器(Container) 容器介绍: docker是通过容器来运行业务的,就像运行一个kvm虚拟机是一样的.容器其实就是从镜像创建的一个实例. 我们可以对容器进行增删改查,容器之间也是相互隔离的.和 ...