概述

JMM规范指出,每一个线程都有自己的工作内存(working memory),当变量的值发生变化时,先更新自己的工作内存,然后再拷贝到主存(main memory),这样其他线程就能读取到更新后的值了。

注意:工作内存和主存是JMM规范里抽象的概念,在JVM的内存模型下,可以将CPU缓存对应作线程工作内存,将JVM堆内存对应主存。

写线程更新后的值何时拷贝到主存?读线程何时从主存中获取变量的最新值?hotspotJVM中引入volatile关键字来解决这些问题,当某个变量被volatile关键字修饰后,多线程对该变量的操作都将直接在主存中进行。在CPU时钟顺序上,某个写操作执行完成后,后续的读操作一定读取的都是最新的值。

内存可见性带来的问题

如下代码片段,写线程每隔1秒递增共享变量counter,读线程是个死循环,如果读线程始终能读取到counter的最新值,那么最终的输出应该是 12345。

public class App {
// 共享变量
static int counter = 0; public static void main(String[] args) {
Thread thread1 = new Thread(() -> {
int temp = 0;
while (true) {
if (temp != counter) {
temp = counter;
// 打印counter的值,期望打印 12345
System.out.print(counter);
}
}
}); Thread thread2 = new Thread(() -> {
for (int i = 0; i < 5; i++) {
counter++;
// 等待1秒,给读线程足够的时间读取变量counter的最新值
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
} // 退出程序
System.exit(0);
}); thread1.start();
thread2.start();
}
}

在没有volatile的情况下,实际的输出结构如下:

1

Process finished with exit code 0

通过volatile解决问题

将共享变量用volatile关键字修饰即可,如下:

// 共享变量
static volatile int counter = 0;

再次执行程序,输出结果如下:

12345

Process finished with exit code 0

综上,volatile关键字使得各个线程对共享变量的操作变得一致。在非volatile字段上做更新操作时,无法保证其修改后的值何时从工作内存(CPU缓存)刷新到主存。对于非volatile字段的读操作也是如此,无法保证线程何时从主存中读取最新的值。

volatile无法保证线程安全性

如下代码片段,多个线程同时递增一个计数器:

public class App {
// 共享变量
static volatile int counter = 0; public static void main(String[] args) throws InterruptedException {
Thread thread1 = new Thread(() -> {
for (int i = 0; i < 10000; i++) {
counter++;
}
}); Thread thread2 = new Thread(() -> {
for (int i = 0; i < 10000; i++) {
counter++;
}
}); thread1.start();
thread2.start();
thread1.join();
thread2.join(); System.out.println("总和:" + counter);
}

输入结果:

总和:12374

如果volatile能保证线程安全,那么输出结果应该是20000,但上面的代码输出12374,所以说,volatile不能解决线程安全(thread)的问题。

所以,还是要通过其他手段来解决多线程安全的问题,比如synchronized。

volatile和synchronized的区别

在上述的代码示例中,我们并没有涉及到多线程竞态(race condition)的问题,核心点是“多线程情况下,对共享变量的写入如何被其他线程及时读取到”。

synchronized关键字是Java中最常用的锁机制,保证临界区(critical section)中的代码在同一个时间只能有一个线程执行,临界区中使用的变量都将直接从主存中读取,对变量的更新也会直接刷新到主存中。所以利用synchronized也能解决内存可见性问题。

代码如下:

public class App {
// 共享变量
static int counter = 0; public static void main(String[] args) {
// 读取变量的线程
Thread readThread = new Thread(() -> {
int temp = 0;
while (true) {
synchronized (App.class) {
if (temp != counter) {
temp = counter;
// 打印counter的值,期望打印 12345
System.out.print(counter);
}
}
}
}); // 修改变量的线程
Thread writeThread = new Thread(() -> {
for (int i = 0; i < 5; i++) {
synchronized (App.class) {
counter++;
} // 等待1秒,给读线程足够的时间读取变量counter的最新值
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
} System.exit(0);
}); readThread.start();
writeThread.start();
}
}

运行,输入结果:

12345

Process finished with exit code 0

虽然通过synchronized也能解决内存可见性的问题,但是这个解决方案也带来了其他问题,比如性能会比较差。

总结

多线程可以提升程序的运行速度,充分利用多核CPU的算力,但多线程也是“恶魔”,会给程序员带来很多问题,比如本文中的内存可见性问题。volatile可以使变量的更新及时刷新到主存,变量的读取也是直接从主存中获取,保证了数据的内存一致性。但是volatile不是用来解决线程安全问题的,无法替代锁机制。

参考:

[1] Java Memory Model - Visibility problem, fixing with volatile variable

[2] Guide to the Volatile Keyword in Java

[3] Managing volatility

[4] Java Volatile Keyword

[5] Thread and Locks

Java内存可见性volatile的更多相关文章

  1. 从一个小例子引发的Java内存可见性的简单思考和猜想以及DCL单例模式中的volatile的核心作用

    环境 OS Win10 CPU 4核8线程 IDE IntelliJ IDEA 2019.3 JDK 1.8 -server模式 场景 最初的代码 一个线程A根据flag的值执行死循环,另一个线程B只 ...

  2. 一个Java内存可见性问题的分析

    如果熟悉Java并发编程的话,应该知道在多线程共享变量的情况下,存在“内存可见性问题”: 在一个线程中对某个变量进行赋值,然后在另外一个线程中读取该变量的值,读取到的可能仍然是以前的值: 这里并非说的 ...

  3. java内存模型-volatile

    volatile 的特性 当我们声明共享变量为 volatile 后,对这个变量的读/写将会很特别.理解 volatile 特性的一个好方法是:把对 volatile 变量的单个读/写,看成是使用同一 ...

  4. 深入理解Java内存模型 - volatile

    volatile的特性 当我们声明共享变量为volatile后,对这个变量的读/写将会很特别.理解volatile特性的一个好方法是:把对volatile变量的单个读/写,看成是使用同一个监视器锁对这 ...

  5. 深入理解Java内存模型——volatile

    volatile的特性 当我们声明共享变量为volatile后,对这个变量的读/写将会非常特别. 理解volatile特性的一个好方法是:把对volatile变量的单个读/写,看成是使用同一个监视器锁 ...

  6. Java内存可见性

    如果一个线程对共享变量的修改,能够被其它线程看到,那么就能说明共享变量在线程之间是可见的.如果一个变量在多个线程的工作内存中都存在副本,那么这个变量就是这几个线程的共享变量.Java内存模型(Java ...

  7. 并发编程之 Java 内存模型 + volatile 关键字 + Happen-Before 规则

    前言 楼主这个标题其实有一种作死的味道,为什么呢,这三个东西其实可以分开为三篇文章来写,但是,楼主认为这三个东西又都是高度相关的,应当在一个知识点中.在一次学习中去理解这些东西.才能更好的理解 Jav ...

  8. Java内存模型-volatile的内存语义

    一 引言 听说在Java 5之前volatile关键字备受争议,所以本文也不讨论1.5版本之前的volatile.本文主要针对1.5后即JSR-133针对volatile做了强化后的了解. 二 vol ...

  9. java 内存可见性

    java线程 -> 线程工作内存 -> 主物理内存 线程工作内存的原理是栈内是连续的小空间,寻址速度比堆快得多,将变量拷贝到栈内生成副本再操作 什么是重排序 代码指令可能并不是严格按照代码 ...

随机推荐

  1. Canny检测算法与实现

    1.原理 图象边缘就是图像颜色快速变化的位置,对于灰度图像来说,也就是灰度值有明显变化的位置.图像边缘信息主要集中在高频段,图像锐化或检测边缘实质就是高通滤波.数值微分可以求变化率,在图像上离散值求梯 ...

  2. 网址封锁的几种方法 公司把 pan.baidu.com 封了 研究实现原理

    HTTP 和 HTTPS 协议HTTP 协议在 头部会发送 host 就是要访问的域名,可以用来被检测. HTTPS 协议虽然会加密全部通讯,但是在握手之前还是明文传输.有证书特证可被检测. 1, D ...

  3. Numpy之数据保存与读取

      在pandas使用的25个技巧中介绍了几个常用的Pandas的使用技巧,不少技巧在机器学习和深度学习方面很有用处.本文将会介绍Numpy在数据保存和读取方面的内容,这些在机器学习和深度学习方向也大 ...

  4. 树莓派上搭建唤醒词检测引擎 Snowboy

    Snowboy 是一款高度可定制的唤醒词检测引擎,可以用于实时嵌入式系统,并且始终监听(即使离线).当前,它可以运行在 Raspberry Pi.(Ubuntu)Linux 和 Mac OS X 系统 ...

  5. CouchDB的简单使用

    一.安装CouchDB 到官网下载CouchDB,在windows下安装CouchDB较为简单,略过. 安装完后,确认CouchDB在运行,然后在浏览器访问http://127.0.0.1:5984/ ...

  6. Redis使用指南

    原文链接 能坚持别人不能坚持的,才能拥有别人未曾拥有的.关注编程大道公众号,让我们一同坚持心中所想,一起成长!! 设置过期时间.释放资源 使用Redis做K-V存储,一定要注意过期时间的把控,任何K- ...

  7. js函数的三种成创建方式以及它们各自的不同

    js有三种创建函数的方式: 1.function语句(也叫函数声明) function sum(a, b) { return a + b; } sum(1, 2); // 3 2. 函数直接量,又叫函 ...

  8. 深入理解JS引擎的执行机制

    深入理解JS引擎的执行机制 1.灵魂三问 : JS为什么是单线程的? 为什么需要异步? 单线程又是如何实现异步的呢? 2.JS中的event loop(1) 3.JS中的event loop(2) 4 ...

  9. js 实现简易留言板功能

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  10. 记 2020蓝桥杯校内预选赛(JAVA组) 赛后总结

    目录 引言 结果填空 1. 签到题 2. 概念题 3. 签到题 4. 签到题 程序题 5. 递增三元组[遍历] 6. 小明的hello[循环] 7. 数位递增[数位dp] 8. 小明家的草地[bfs] ...