要想学好JUC,还得先了解 volatile 这个关键字。了解 volatile ,我们从一个例子开始吧。

本文不会很详细去说java内存模型,只是很简单地学习一下volatile

一个例子

package jfound.demo;
import java.util.concurrent.TimeUnit;
public class TaskRunner {
private static boolean ready = true;
public static void main(String[] args) throws InterruptedException {
new Thread(() -> {
while (ready) {
}
}).start();
TimeUnit.SECONDS.sleep(1);
ready = false;
}
}

这个程序里面,新开一个线程,ready 初始化值为true, 线程里面是一个死循环,当 ready 修改为 false的时候,我们希望线程里面的死循环会结束,然后jvm会停止。

然后在这个例子里面,程序根本不会停止。但当 readyvolatile 关键字修饰的时候,程序符合我们预期,停止了。

....
private static volatile boolean ready = true;
...

CPU执行及缓存

CPU负责执行程序指令,但是他们需要从内存(RAM)中获取程序指令和所需要的数据。由于CPU每秒能执行大量的执行,如果每执行一次指令就从内存(RAM)中获取数据的话,显然是不够理想的,毕竟CPU与内存之间还是有一定的距离的。为了改善这种情况,CPU会有一系列的优化,例如指令重排序,当然,还有缓存。下图为CPU及内存层次的结构。

当CPU获取指令的时候,也会把指令所需要的数据读进CPU缓存中,当在某些时刻,通常是指令改变或者缓存失效时,CPU会重新从内存(RAM)中读取指令或数据。

在上面的例子中,新开的线程在做循环的时候,会读取 ready 变量到该线程所执行的CPU缓存中,当 main 线程修改 ready 变量为 false 的时候,是首先写在 main 线程所执行的CPU的缓存中,在某些时刻才会写入到内存(RAM)中。也就是说要让新开的线程停止的话,必须是 main 线程修改的变量写入到内存(RAM)中,而且新开的线程的所在的CPU缓存要失效,让其重新读取 ready 变量。然而,没有加 volatile 之前,main 线程并不会实时把变量 ready 写入到内存(RAM)中去,新开的线程也不会从内存中获取 ready 新的数据。

缓存一致性协议(MESI协议)

上述的问题就是大名鼎鼎的缓存不一致性的问题,也就是在并发编程中所要解决的主要问题之一。

在早期的CPU中,是通过在总线(上图中的Bus)上加LOCK#的形式来解决缓存不一致的问题,当加上总线锁的时候,加锁的CPU就独占内存,其他CPU就不能读取内存,也就是不能执行指令,只能乖乖等待锁释放,这样的总线锁效率很低,不过是能解决了缓存不一致的问题。

为了提高效率,就出现了缓存一致性协议。缓存一致性是为了保证每个缓存中使用的共享变量的副本是一致的,它的核心思想是:当CPU写数据时,如果发现该操作的变量是共享变量,即使在其他CPU中也存在该变量的副本,会发出通知,让其他CPU该变量的缓存行置为无效状态,因此即使其他CPU需要读取这个变量时,发现自己缓存中的该变量的缓存行无效了,那么就会从内存中重新读取。

MESI全名是Modified、Exclusive、 Share or Invalid,使每一个缓存行可能处于M、E、S和I这四种状态之一,

  • M:被修改的。处于这一状态的数据,只在本CPU中有缓存数据,而其他CPU中没有。同时其状态相对于内存中的值来说,是已经被修改的,且没有更新到内存中。
  • E:独占的。处于这一状态的数据,只有在本CPU中有缓存,且其数据没有修改,即与内存中一致。
  • S:共享的。处于这一状态的数据在多个CPU中都有缓存,且与内存一致。
  • I:无效的。本CPU中的这份缓存已经无效。

例子解析

volatile 关键字有着上面所说的触发缓存一致性的功能,所以在加上 volatile 关键字之后,main 线程把 ready 修改为 false 的时候,新开的线程是可以读取到修改后的 ready 的值,所以程序是可以符合我们的预期,停止了。

总结

本文通过上面的一个小例子来解析了 volatile 的一个功能,缓存一致性,为接下来学习 JUC 做准备。当然 volatile 关键字在java中还会有其他的功能,例如 happer-before、内存屏障、重排序等等,这些就不在本文赘述了。

微信关注我,发现更多java领域知识

JUC整理笔记二之聊聊volatile的更多相关文章

  1. JUC整理笔记一之细说Unsafe

    JUC(java.util.concurrent)的开始,可以说是从Unsafe类开始. Unsafe 简介 Unsafe在sun.misc 下,顾名思义,这是一个不安全的类,因为Unsafe类所操作 ...

  2. JUC整理笔记三之测试工具jcstress

    并发测试工具Jcstress使用教程 Jcstress 全称 Java Concurrency Stress,是一种并发压力测试工具,可以帮助研究JVM.java类库和硬件中并发的正确性. Wiki地 ...

  3. jQuery整理笔记文件夹

    jQuery整理笔记文件夹 jQuery整理笔记一----jQuery開始 jQuery整理笔记二----jQuery选择器整理 jQuery整理笔记三----jQuery过滤函数 jQuery整理笔 ...

  4. JUC学习笔记(二)

    JUC学习笔记(一)https://www.cnblogs.com/lm66/p/15118407.html 1.Lock接口 1.1.Synchronized 1.1.1.Synchronized关 ...

  5. canvas学习之API整理笔记(二)

    前面我整理过一篇文章canvas学习之API整理笔记(一),从这篇文章我们已经可以基本了解到常用绘图的API.简单的变换和动画.而本篇文章的主要内容包括高级动画.像素操作.性能优化等知识点,讲解每个知 ...

  6. JUC学习笔记(四)

    JUC学习笔记(一)https://www.cnblogs.com/lm66/p/15118407.html JUC学习笔记(二)https://www.cnblogs.com/lm66/p/1511 ...

  7. xmpp整理笔记:发送图片信息和声音信息

    图片和音频文件发送的基本思路就是: 先将图片转化成二进制文件,然后将二进制文件进行base64编码,编码后成字符串.在即将发送的message内添加一个子节点,节点的stringValue(节点的值) ...

  8. xmpp整理笔记:聊天信息的发送与显示

    任何一个信息的发送都需要关注两个部分,信息的发出,和信息在界面中的显示 往期回顾: xmpp整理笔记:环境的快速配置(附安装包)  http://www.cnblogs.com/dsxniubilit ...

  9. xmpp整理笔记:用户网络连接及好友的管理

    xmpp中的用户连接模块包括用户的上线与下线信息展现,用户登录,用户的注册: 好友模块包括好友的添加,好友的删除,好友列表的展示. 在xmpp中 负责数据传输的类是xmppStream,开发的过程中, ...

随机推荐

  1. Leetcode PHP题解--D75 706. Design HashMap

    2019独角兽企业重金招聘Python工程师标准>>> D75 706. Design HashMap 题目链接 706. Design HashMap 题目分析 自行设计一个has ...

  2. 解决layui动态追加的点击事件不起作用问题

    2019独角兽企业重金招聘Python工程师标准>>> //不起作用 $('#demo').on('click', function() { layer.msg('响应点击事件'); ...

  3. 微信Webapp开发的各种变态路由需求及解决办法!

    前言 最近在使用BUI Webapp开发的一个小商城项目在微信上遇到一些坑及变态需求, 层层深入, 整理一下给后来人参考. 一定有你还不知道的! 调试缓存 问题描述: 微信打开的web页面默认是会缓存 ...

  4. SQL Server遍历表中记录的2种方法

    SQL Server遍历表一般都要用到游标,SQL Server中可以很容易的用游标实现循环,实现SQL Server遍历表中记录.本文将介绍利用使用表变量和游标实现数据库中表的遍历. 表变量来实现表 ...

  5. CF--思维练习--CodeForces - 220C Little Elephant and Shifts (STL模拟)

    ACM思维题训练集合 The Little Elephant has two permutations a and b of length n, consisting of numbers from ...

  6. 积性函数初步(欧拉$\varphi$函数)

    updata on 2020.4.3 添加了欧拉\(\varphi\)函数为积性函数的证明和它的计算方式 1.积性函数 设\(f(n)\)为定义在正整数上的函数,若\(f(1)=1\),且对于任意正整 ...

  7. 题目分享L

    题意:n个人围成一个环,每个人初始有一些金币,每个人可以把金币递给相邻的人,问最少传递多少金币使每个人金币数相同? 分析:首先在保证最优的情况下不可能会出现相邻的两个人互相送金币,因为这样他们公共的部 ...

  8. Scrapy爬虫框架基本使用

    scrapyhub上有些视频简单介绍scrapy如何学习的(貌似要FQ):https://helpdesk.scrapinghub.com/support/solutions/articles/220 ...

  9. Linux服务器有大量的TIME_WAIT状态

    我们经常会遇到在服务器上看到大量的TIME_WAIT,它们占用进程不释放,最后会导致所有进程数被耗完,服务器负载增高等生产事故,具体是什么原因导致的呢?我们先来看看TCP的三次握手四次挥手都是怎样的一 ...

  10. jQuery简单竖排手风琴折叠菜单代码

    项目需求1.刚开始只显示,每个标题, 2.让每个 li列表隔行换色 3.当我点击某个标题时,下面的列表会缓慢的展开,其他列表展开的内容会收起 <!DOCTYPE html> <htm ...