Python 官方文档:入门教程 => 点击学习
目录一、通过程序看现象二、为什么会产生这种现象(JMM模型)?三、MESI 缓存一致性协议 一、通过程序看现象 在开始为大家讲解Java 多线程缓存模型之前,我们先看下面的这一段代码
在开始为大家讲解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也没有跳出死循环**,这是为什么呢?
要解释上面提到的问题,我们就需要学习JMM(Java Memory Model)Java 内存模型,笔者觉得叫做Java多线程内存模型更准确一些。
shareFlag=true
的时候将其工作内存副本修改为shareFlag=true
,同时将副本的值同步写回(store&write)到主内存中。shareFlag=false
没有发生变化,所以线程1一直处于死循环之中。按照上文的实验以及JMM模型,线程2修改的共享变量的值,线程1感知不到。那怎么样才能让线程1感知到共享变量的值发生了变化呢?其实也很简单,给shareFlag共享变量加上volatile关键字就可以了。
public volatile static boolean shareFlag = false;
其底层原理是这样的,加上volatile关键字提示JMM遵循MESI 缓存一致性协议,该协议包含如下的缓存使用规范(看不懂可以不看,下文会用简单的语言及例子描述一下)。
上文中的缓存使用规范可能过于复杂,简单的说就是
到此这篇关于java并发-volatile与JMM多线程内存模型的文章就介绍到这了,更多相关java多线程内存模型内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!
--结束END--
本文标题: Java并发编程之volatile与JMM多线程内存模型
本文链接: https://www.lsjlt.com/news/148645.html(转载时请注明来源链接)
有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341
下载Word文档到电脑,方便收藏和打印~
2024-03-01
2024-03-01
2024-03-01
2024-02-29
2024-02-29
2024-02-29
2024-02-29
2024-02-29
2024-02-29
2024-02-29
回答
回答
回答
回答
回答
回答
回答
回答
回答
回答
0