Java并发编程之volatile与JMM多线程内存模型

作者:字母哥哥 时间:2023-10-19 12:13:48 

Java并发编程之volatile与JMM多线程内存模型

一、通过程序看现象

在开始为大家讲解Java 多线程缓存模型之前,我们先看下面的这一段代码。这段代码的逻辑很简单:主线程启动了两个子线程,一个线程1、一个线程2。线程1先执行,sleep睡眠2秒钟之后线程2执行。两个线程使用到了一个共享变量shareFlag,初始值为false。如果shareFlag一直等于false,线程1将一直处于死循环状态,所以我们在线程2中将shareFlag设置为true

public class VolatileTest {
 public static boolean shareFlag = false;
 public static void main(String[] args) throws InterruptedException {
   new Thread(() -> {
     System.out.print("开始执行线程1 =>");
     while (!shareFlag){  //shareFlag = false则一直死循环
       //System.out.println("shareFlag=" + shareFlag);
     }
     System.out.print("线程1执行完成 =>");
   }).start();
   Thread.sleep(2000);
   new Thread(() -> {
     System.out.print("开始执行线程2 =>");
     shareFlag = true;
     System.out.print("线程2执行完成 =>");
   }).start();
 }
}

如果你没有学过JMM线程模型,可能你看完上面的代码,希望得到的输出结果是下面这样的:

开始执行线程1 =>开始执行线程2 =>线程2执行完成 =>线程1执行完成=>

如下图所示,正常人理解这段代码,首先执行线程1进入循环,线程2修改shareFlag=true,线程1跳出循环。所以跳出循环的线程1会打印"线程1执行完成=>",但是经过笔者实验,**"线程1执行完成=>"不会被打印,线程1也没有跳出死循环**,这是为什么呢?

Java并发编程之volatile与JMM多线程内存模型

二、为什么会产生这种现象(JMM模型)?

要解释上面提到的问题,我们就需要学习JMM(Java Memory Model)Java 内存模型,笔者觉得叫做Java多线程内存模型更准确一些。

Java并发编程之volatile与JMM多线程内存模型

  • 首先,在JMM中每个线程有自己的工作内存,在程序启动的时候,线程将共享变量加载(read&load)到自己的工作内存中,加载到线程工作内存中的内存变量是主内存中共享变量的副本。也就是说此时shareFlag在内存中有三个副本,值都等于false。

  • 当线程2执行shareFlag=true的时候将其工作内存副本修改为shareFlag=true,同时将副本的值同步写回(store&write)到主内存中。

  • 但是线程1的工作内存中的shareFlag=false没有发生变化,所以线程1一直处于死循环之中

三、MESI 缓存一致性协议

按照上文的实验以及JMM模型,线程2修改的共享变量的值,线程1感知不到。那怎么样才能让线程1感知到共享变量的值发生了变化呢?其实也很简单,给shareFlag共享变量加上volatile关键字就可以了。

public volatile static boolean shareFlag = false;

其底层原理是这样的,加上volatile关键字提示JMM遵循MESI 缓存一致性协议,该协议包含如下的缓存使用规范(看不懂可以不看,下文会用简单的语言及例子描述一下)。

  1. Modified:代表当前Cache行的数据是修改过的(Dirty),并且只在当前CPU的Cache中是修改过的;此时该Cache行的数据与其他Cache中的数据不同,与内存中该行的数据也不同。

  2. Exclusive:代表当前Cache行的数据是有效数据,其他CPU的Cache中没有这行数据;并且当前Cache行数据与内存中的数据相同。

  3. Shared:代表多个CPU的Cache中都会缓存有这行数据,并且Cache中的数据与内存中的数据一致;

  4. Invalid:表示当前Cache行中的数据无效;

Java并发编程之volatile与JMM多线程内存模型

上文中的缓存使用规范可能过于复杂,简单的说就是

  • 当线程2修改shareFlag的时候(参考Modify),告知bus总线我修改了共享变量shareFlag,

  • 线程1对Bus总线进行监听,当它获知共享变量shareFlag发生了修改就会将自己工作内存中的shareFlag副本删除使其失效。

  • 当线程1再次需要使用到shareFlag的时候,发现工作内存中没有shareFlag变量副本,就会重新从主内存中加载(read&load)

来源:https://blog.csdn.net/hanxiaotongtong/article/details/124723177

标签:java,多线程,内存,模型
0
投稿

猜你喜欢

  • c#使用Dataset读取XML文件动态生成菜单的方法

    2022-07-14 22:52:20
  • SWT(JFace)体验之模拟BorderLayout布局

    2022-08-17 18:09:51
  • 使用Spring的拦截器监测每个Controller或方法的执行时长

    2021-12-19 16:36:01
  • Spring Security基于散列加密方案实现自动登录功能

    2022-09-19 01:10:22
  • C++封装静态链接库和使用的详细步骤

    2021-08-28 00:54:13
  • C#动态webservice调用接口

    2023-10-18 07:22:00
  • android通过Location API显示地址信息的实现方法

    2021-09-16 15:41:02
  • C语言实现学生信息管理系统

    2023-05-24 12:08:50
  • mybatis根据表逆向自动化生成代码的实现

    2023-03-12 15:39:02
  • C++实现优先队列的示例详解

    2022-04-06 22:14:20
  • JAVA熔断和降级真实关系的图文详解

    2023-11-30 12:39:20
  • 详解Java中String类的各种用法

    2022-03-10 12:48:35
  • Unity实现枚举类型中文显示

    2023-02-22 12:00:28
  • springboot中的pom文件 project报错问题

    2022-01-24 00:41:55
  • Android开发-之五大布局详解

    2021-11-30 14:24:30
  • 一文带你搞懂Java中的递归

    2022-10-08 07:34:04
  • flutter实现头部tabTop滚动栏

    2022-03-21 14:00:06
  • Android之FanLayout制作圆弧滑动效果

    2023-01-14 16:58:29
  • android使用gesturedetector手势识别示例分享

    2023-08-08 15:06:11
  • 一篇文章弄懂Spring MVC的参数绑定

    2023-09-17 01:01:21
  • asp之家 软件编程 m.aspxhome.com