JUC(一):volatile关键字
volatile是什么
volatile是java虚拟机提供的轻量级同步机制,它包含三种特性:
- 保证可见性:只要主内存中变量做出修改,其余线程马上会感知到变量的修改。
package com.chinda.java.audition;
import java.util.concurrent.TimeUnit;
class MyData {
volatile int number = 0;
public void addTo60 (){
this.number = 60;
}
}
/**
* 0. 先不添加volatile关键字, 执行是否会阻塞; 之后加上volatile关键字, 最后打印会不会执行。
* 1.验证volatile可见性
* 1.1 假如int number = 0; number变量没有添加volatile关键词修饰没有可见性
* 1.2 添加volatile,可以解决可见性问题。
*
* @author Wang Chinda
* @date 2020/5/2
* @see
* @since 1.0
*/
public class VolatileDemo {
public static void main(String[] args) {
MyData myData = new MyData();
new Thread(() -> {
System.out.println(Thread.currentThread().getName() + "\t come in!");
try {
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
myData.addTo60();
System.out.println(Thread.currentThread().getName() + "\t update number value: " + myData.number);
}, "T1").start();
while (myData.number == 0) {
// 主线程跳出此循环, 说明线程时可见, 若阻塞在此循环, 说明线程不可见。
}
System.out.println(Thread.currentThread().getName() + "\t mission is over! numer value: " + myData.number);
}
}
- 不保证原子性:
package com.chinda.java.audition;
import java.util.concurrent.atomic.AtomicInteger;
class MyData1 {
volatile int number = 0;
AtomicInteger atomic = new AtomicInteger();
public void addPlus() {
this.number++;
}
public void addAtomic() {
atomic.getAndIncrement();
}
}
/**
* 1.验证volatile不保证原子性
* 1.1 原子性: 即某个线程正在做某个具体业务, 中间不可以被加塞或者被分割。需要整体完整,要么同时成功,要么同时失败。
* 1.2 一个线程将属性值写回主内存,没有来得及通知其他线程时, 有其他线程也要将属性值写回主内存, 属性值在主内存中出现覆盖现象。所以volatile不保证原子性。
* 2. 解决非原子性问题。
* 2.1 添加synchronized
* 2.2 添加锁lock
* 2.3 用原子类AtomicInteger
*
* @author Wang Chinda
* @date 2020/5/2
* @see
* @since 1.0
*/
public class VolatileDemo1 {
public static void main(String[] args) {
MyData1 myData = new MyData1();
for (int i = 1; i <= 20; i++) {
new Thread(() -> {
for (int j = 0; j < 1000; j++) {
myData.addPlus();
myData.addAtomic();
}
}, String.valueOf(i)).start();
}
// 等待上面20个线程全部计算完成后, 再用main线程取得最终的计算结果。
while (Thread.activeCount() > 2) {
Thread.yield();
}
System.out.println(Thread.currentThread().getName() + "\t finally! numer value: " + myData.number);
System.out.println(Thread.currentThread().getName() + "\t finally! atomic value: " + myData.atomic.get());
}
}
- 禁止指令重排
JMM概念
JMM本身是一种抽象的概念,并不真实存在,它描述的是一组规则或规范,通过这组规范定义了程序中各个变量(实例字段、静态字段、构成数组对象的元素)的访问方式。包含三种特性:
- 可见性
- 原子性
- 有序性
JMM同步规定
线程解锁前,必须把共享变量的值刷回主内存。
线程加锁前,必须读取主内存内最新值到线程的工作内存中。
加锁与解锁是同一把锁。
由于JVM运行程序的实体是线程,而每个线程创建时JVM都会为其创建一个工作内存,工作内存时内阁线程私有数据区域,而Java内存模型中规定所有变量都存储在主内存,主内存时共享内存区域,所有线程都可以访问,但线程对变量的操作必须在工作内存中进行,首先要将变量从主内存拷贝到自己的工作内存空间,然后对变量进行操作,操作完成将变量写回主内存。
无volatile修饰JMM

无volatile修饰时,主内存中属性修改,各线程中是不可见的。各个线程中工作内存只以运算前在主内存中拷贝的属性值为准。
volatile修饰JMM

有volatile修饰时,主内存中的属性修改,各个线程中是可见的,其中一个线程将工作内存中的属性写回主内存中会通知各个线程去同步主内存中的属性值。但是若在通知之前已经将工作内存中的属性修改,再收到通知同步,这时候就会出现属性值覆盖情况。例如上图中,你品,你细品。即volatile可保证线程间的可见性,但是不保证线程中数据的原子性。
单例模式双端检锁线程安全
DCL(双端检锁)机制不一定线程安全,原因是指令重排序的存在,加入volatile可以禁止指令指令重排。
package com.chinda.java.audition;
/**
* @author Wang Chinda
* @date 2020-02-16 21:27
* @see
* @since 1.0
*/
public class Singleton5 {
private static volatile Singleton5 INSTANCE;
private Singleton5() {
System.out.println("线程: " + Thread.currentThread().getName() + " --> 此行代码被执行多次, 说明线程不安全!!!");
}
public static Singleton5 getInstance() throws InterruptedException {
// 为提升效率
if (INSTANCE == null) {
// 类锁, 线程每次都会抢锁
synchronized (Singleton5.class) {
if (INSTANCE == null) {
Thread.sleep(100);
INSTANCE = new Singleton5();
}
}
}
return INSTANCE;
}
public static void main(String[] args) {
for (int i = 1; i <= 20000; i++) {
new Thread(() -> {
try {
Singleton5.getInstance();
} catch (InterruptedException e) {
e.printStackTrace();
}
}, String.valueOf(i)).start();
}
}
}
原因在于某一个线程执行到第一次检测,读取到INSTANCE不为null时,INSTANCE的引用对象可能没有完成初始化。
INSTANCE = new Singleton5();可以分为一下三步完成。
- memory = allocate(); 1. 分配对象内存空间。
- INSTANCE -> memory ; 2. 初始化对象。
- INSTANCE = ; 3. 设置INSTANCE指向刚分配的内存地址,此时INSTANCE != null。
步骤2和步骤3不存在数据依赖关系,而且无论重排前还是重排后程序的执行结果再单线程中并没有变化,因此这种重排优化时允许的。
- memory = allocate(); 1. 分配对象内存空间。
- INSTANCE = ; 3. 设置INSTANCE指向刚分配的内存地址,此时INSTANCE != null,但对象还没有初始化完成。
- INSTANCE -> memory ; 2. 初始化对象。
指令重排只会保证串行语义执行的一致性(单线程),但是并不会关心多线程间的语义一致性。所有当一个线程访问INSTANCE不为null时,由于INSTANCE实例未必已经初始化完成,也就造成了线程安全问题。
JUC(一):volatile关键字的更多相关文章
- volatile关键字与内存可见性&原子变量与CAS算法
1 .volatile 关键字:当多个线程进行操作共享数据时, 可以保证内存中的数据可见 2 .原子变量:jdk1.5后java.util.concurrent.atomic 包下提供常用的原子变量 ...
- 【JUC系列第一篇】-Volatile关键字及内存可见性
作者:毕来生 微信:878799579 什么是JUC? JUC全称 java.util.concurrent 是在并发编程中很常用的实用工具类 2.Volatile关键字 1.如果一个变量被volat ...
- JUC 并发编程--05, Volatile关键字特性: 可见性, 不保证原子性,禁止指令重排, 代码证明过程. CAS了解么 , ABA怎么解决, 手写自旋锁和死锁
问: 了解volatile关键字么? 答: 他是java 的关键字, 保证可见性, 不保证原子性, 禁止指令重排 问: 你说的这三个特性, 能写代码证明么? 答: .... 问: 听说过 CAS么 他 ...
- Volatile关键字以及线程的内存可见性问题
一.Volatile关键字 作用: 当多个线程进行操作共享数据时,可以保证内存中的数据可见,即为一个线程对数据的修改对另外一个线程来说是可见的.相较于 synchronized 是一种较为轻量级的同步 ...
- volatile关键字与内存可见性
前言 首先,我们使用多线程的目的在于提高程序的效率,但是如果使用不当,不仅不能提高效率,反而会使程序的性能更低,因为多线程涉及到线程之间的调度.CPU上下文的切换以及包括线程的创建.销毁和同步等等,开 ...
- volatile关键字及内存可见性
先看一段代码: package com.java.juc; public class TestVolatile { public static void main(String[] args) { T ...
- Java volatile关键字小结
public class Test { public static void main(String[] args){ } } /* 12.3 Java内存模型 Java内存模型定义了线程与主内存之间 ...
- 1.volatile关键字 内存可见性
Java JUC 简介 在 Java 5.0 提供了 java.util.concurrent (简称JUC )包,在此包中增加了在并发编程中很常用的实用工具类,用于定义类似于线程的自定义子系统,包括 ...
- 最后一面挂在volatile关键字上,面试官:重新学学Java吧!
最后一面挂在volatile关键字上,面试官:重新学学Java吧! 为什么会有volatile关键字? volatile: 易变的; 无定性的; 无常性的; 可能急剧波动的; 不稳定的; 易恶化的; ...
随机推荐
- 信息收集之——旁站、C段
旁站的概念 旁站指的是同一服务器上的其他网站,很多时候,有些网站可能不是那么容易入侵.那么,可以查看该网站所在的服务器上是否还有其他网站.如果有其他网站的话,可以先拿下其他网站的webshell,然 ...
- Appium上下文和H5测试(一)
坚持原创输出,点击蓝字关注我吧 作者:清菡 博客:oschina.云+社区.知乎等各大平台都有. 目录 一.混合应用-H5 1.混合应用是什么? 2.怎么样分辨一个 App 页面究竟是原生的还是 We ...
- 移动端调试Web页面
移动端调试Web页面 虽然可以在PC下,通过开发者工具,模拟移动端,但是这样只能模拟页面样式,对于代码的执行情况是无法模拟的,所以在此结合实际调试经验,针对安卓与IOS设备,进行总结. IOS 安卓 ...
- win10安装jenkins忘记密码的解决方法
jenkins安装完了一直没用,突然想学习的时候,忘记了登陆密码. 一:修改配置文件 1. 打开jenkins的安装目录,选择users下面的admin目录下的config.xml文件 我的文件路 ...
- redis cluster可用性测试
上一节,我们用三台redis组成了cluster,现在我们停掉一台试试: 比较奇怪的是,在停掉其中一台服务器之前建立的链接仍然可以正常执行命令,当我们断开重连时,命令就都被拒绝了: 关联知识: 什么时 ...
- dubbo协议之响应头编码器&响应对象编码
前2节分析完了请求头和请求对象的编码,这里看一下响应头和响应对象的编码: 和请求头部一样进来先指定序列化器,没有的话用默认的Hessian2,接下来2个字节的操作和请求头编码类似,第三个字节时去req ...
- 【数据结构】关于前缀树(单词查找树,Trie)
前缀树的说明和用途 前缀树又叫单词查找树,Trie,是一类常用的数据结构,其特点是以空间换时间,在查找字符串时有极大的时间优势,其查找的时间复杂度与键的数量无关,在能找到时,最大的时间复杂度也仅为键的 ...
- linux ssh远程连接控制 linux(centOS) 口令、密钥连接
sshd服务提供两种安全验证的方法: 基于口令的安全验证:经过验证帐号与密码即可登陆到远程主机. 基于密钥的安全验证:需要在本地生成"密钥对"后将公钥传送至服务端,进行公共密钥的比 ...
- Java 关于策略模式+简单工厂模式下的思考
导读 最近在做公司一个消息网关的服务,包括:短信.微信.邮件等,所有请求通过一个入口,方便接口的管理(记录日志.接口限流白名单啥的).如何写这个接口呢,还有为了以后扩展,对接过短信.微信.公众号的童鞋 ...
- Guava中EventBus分析
EventBus 1. 什么是EventBus 总线(Bus)一般指计算机各种功能部件之间传送信息的公共通信干线,而EventBus则是事件源(publisher)向订阅方(subscriber)发送 ...