java学习:JMM(java memory model)、volatile、synchronized、AtomicXXX理解
一、JMM(java memory model)内存模型
从网上淘来二张图:

上面这张图说的是,在多核CPU的系统中,每个核CPU自带高速缓存,然后计算机主板上也有一块内存-称为主内(即:内存条)。工作时,CPU的高速缓存中的数据通过一系列手段来保证与主内的数据一致(CacheCoherence),更直白点,高速缓存要从主内中load数据,处理完以后,还要save回主存。

上图说的是,java中的各种变量(variable)保存在主存中,然后每个线程自己也有自己的工作内存区(working memory),工作时,线程从主存中把变量副本load到自己的工作内存区,处理完了,再save回主存。
好象很明白,没有什么不好理解的:),
问题来了,如果有二个线程:线程A与线程B, A从主存中读取了变量x(到自己的的工作内存区),正准备处理,这时B修改了主存中的变量x,线程A能看见这种变化吗?(是否需要及时从主存中,加载最新的值),这个问题称为共享变量的可见性。
二、volatile、synchronized、AtomicXXX
直接上码:
2.1 版本1
package test.cn.mwee.order.monitor; /**
* Created by 菩提树下的杨过 on 2017/6/11.
*/
public class ThreadTest extends Thread { private static boolean flag = false; public void run() {
System.out.println("t1:" + Thread.currentThread().getId());
while (!flag) { }
System.out.println("quit!");
} public static void main(String[] args) throws InterruptedException {
ThreadTest t1 = new ThreadTest();
t1.start();
Thread.sleep(50);
ThreadTest.flag = true;
System.out.println("main:" + Thread.currentThread().getId());
}
}
ThreadTest是一个线程类,里面有一个静态变量flag,然后写了个main方法做测试。
注:在t1启动完成后,主线程中修改了ThreadTest的静态变量值flag,这时t1的run方法里的while循环,其实是看不见主线程对这个值的修改,所以程序始终不能退出,打印不出那一行quit.
2.2 版本2
package test.cn.mwee.order.monitor; /**
* Created by 菩提树下的杨过 on 2017/6/11.
*/
public class ThreadTest extends Thread { private static boolean flag = false; public void run() {
System.out.println("t1:" + Thread.currentThread().getId());
while (!flag) {
synchronized (Class.class) {
}
}
System.out.println("quit!");
} public static void main(String[] args) throws InterruptedException {
ThreadTest t1 = new ThreadTest();
t1.start();
Thread.sleep(50);
ThreadTest.flag = true;
System.out.println("main:" + Thread.currentThread().getId());
}
}
相对版本1,while循环中增加了一个synchronized同步代码块,虽然里面啥代码也没有,但是再次运行,能正常quit了(想下为啥?)
答案:(也是从网上抄来的)
synchronized关键字强制实现一个互斥锁,使得被保护的代码块在同一时间只能有一个线程进入并执行。同时也带另外一个作用:在线程进入synchronized块之前,会把工作存内存中的所有内容映射到主内存上,然后把工作内存清空再从主存储器上拷贝最新的值。而在线程退出synchronized块时,同样会把工作内存中的值映射到主内存,但此时并不会清空工作内存。这样,工作内存中的值和主内存中的值是一致的,保证了数据的一致性!(换句话说,run中的while看到了主存中的flag变量值的改变)
思考题:如果在while{}里写一行println打印输出,即:
while (!flag) {
System.out.println("flag=" + flag);
}
也能正常退出,留给大家去想。(提示:可以去看看println的源码实现)
2.3 版本3
package test.cn.mwee.order.monitor; /**
* Created by 菩提树下的杨过 on 2017/6/11.
*/
public class ThreadTest extends Thread { private volatile static boolean flag = false; public void run() {
System.out.println("t1:" + Thread.currentThread().getId());
while (!flag) { }
System.out.println("quit!");
} public static void main(String[] args) throws InterruptedException {
ThreadTest t1 = new ThreadTest();
t1.start();
Thread.sleep(50);
ThreadTest.flag = true;
System.out.println("main:" + Thread.currentThread().getId());
}
}
相对版本1,flag变量前加了关键字volatile,它能保证对该变量的修改,同步到其它线程,即其它线程读取flag时,看到的就是变化后的最新值,同时volatile还能防止指令重排序。运行下,也能如期打印出quit,程序退出。
注:volatile只能保证其它线程看到的变量值是最新的,但是并不保证原子性(换句话说,高并发情况下,仍然无法100%保证线程安全)
2.4 版本4
package test.cn.mwee.order.monitor; import java.util.concurrent.atomic.AtomicBoolean; /**
* Created by 菩提树下的杨过 on 2017/6/11.
*/
public class ThreadTest extends Thread { private static AtomicBoolean flag = new AtomicBoolean(false); public void run() {
System.out.println("t1:" + Thread.currentThread().getId());
while (!flag.get()) { }
System.out.println("quit!");
} public static void main(String[] args) throws InterruptedException {
ThreadTest t1 = new ThreadTest();
t1.start();
Thread.sleep(50);
ThreadTest.flag.set(true);
System.out.println("main:" + Thread.currentThread().getId());
}
}
与上一个版本相比,使用了可以保证原子性,又不用加同步锁的并发包里的AtomicXXX系列类,同样也可以正常打印出quit,推荐使用这个。
如果感兴趣的话,可以看下AtomicBoolean的源码,其实是借助volatile以及CAS来实现的,源码一看便知,不再啰嗦。
java学习:JMM(java memory model)、volatile、synchronized、AtomicXXX理解的更多相关文章
- [java学习笔记]java语言核心----面向对象之this关键字
一.this关键字 体现:当成员变量和函数的局部变量重名时,可以使用this关键字来区别:在构造函数中调用其它构造函数 原理: 代表的是当前对象. this就是所在函数 ...
- [java学习笔记]java语言核心----面向对象之构造函数
1.构造函数概念 特点: 函数名与类名相同 不用定义返回值类型 没有具体的返回值 作用: 给对象进行初始化 注意: 默认构造函数 多个构造函数是以重载出现的 一个类中如果 ...
- [ Java学习基础 ] Java构造函数
构造方法是类中特殊方法,用来初始化类的实例变量,它在创建对象(new运算符)之后自动调用. Java构造方法的特点如下: 构造方法名必须与类名相同. 构造方法没有任何返回值,包括void. 构造方法只 ...
- [ Java学习基础 ] Java的继承与多态
看到自己写的东西(4.22的随笔[ Java学习基础 ] Java构造函数)第一次达到阅读100+的成就还是挺欣慰的,感谢大家的支持!希望以后能继续和大家共同学习,共同努力,一起进步!共勉! ---- ...
- [ Java学习基础 ] Java的抽象类与接口
一.抽象类 1. 抽象类 Java语言提供了两种类:一种是具体类:另一种是抽象子类. 2. 抽象类概念: 在面向对象的概念中,所有的对象都是通过类来描绘的,但是反过来,并不是所有的类都是用来描绘对象的 ...
- Java 线程 — JMM Java内存模型
JMM Java Memory Model,Java内存模型,属于语言级的内存模型 并发编程中存在的问题: 如何通信:用于线程之间交换信息.两种方式:共享内存,消息传递 如何同步:用于控制不同线程间操 ...
- Java学习笔记 -- Java定时调度工具Timer类
1 关于 (时间宝贵的小姐姐请跳过) 本教程是基于Java定时任务调度工具详解之Timer篇的学习笔记. 什么是定时任务调度 基于给定的时间点,给定的时间间隔或者给定的执行次数自动执行的任务. 在Ja ...
- 【Java学习笔记之二十六】深入理解Java匿名内部类
在[Java学习笔记之二十五]初步认知Java内部类中对匿名内部类做了一个简单的介绍,但是内部类还存在很多其他细节问题,所以就衍生出这篇博客.在这篇博客中你可以了解到匿名内部类的使用.匿名内部类要注意 ...
- java学习路线图-----java基础学习路线图(J2SE学习路线图)
安装JDK和开发软件跳过,网上太多了,不做总结,以下是我总结的学习路线图,欢迎补充. JAVA基础语法 注释,标识符命名规则及Java中的关键字 Java基本数据类型 Java运算符与表达式 Java ...
- Java学习之Java接口回调理解
Java接口回调 在Java学习中有个比较重要的知识点,就是今天我们要讲的接口回调.接口回调的理解如果解释起来会比较抽象,我一般喜欢用一个或几个经典的例子来帮助加深理解. 举例:老板分派给员工做事,员 ...
随机推荐
- 在ASP.Net中两种利用CSS实现多界面的方法
通过使页面动态加载不同CSS实现多界面(类型于csdn的blog): 方法一: <%@page language="C#"%><%@import namespac ...
- ASP.NET MVC3-Music Store中英文教程 [下载]
翻译原文档名: MVC Music Store版本: ASP.NET MVC3概述Mvc Music Store 是一个为WEB开发人员一步一步介绍和解释如何使用MVC和Visual Web开发的应用 ...
- 经典设计模式-iOS的实现
最近看了<HeadFirst 设计模式>这本书,给组内伙伴准备一次分享,把这次分享记录下来,有需要的可以看看. 这本书主要介绍了四人帮23种经典设计模式中的的14种,也是常用的几种.看完这 ...
- HDU 2054 又见GCD
又见GCD Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Subm ...
- HDU 3787 A+B 模拟题
解题报告:就是输入两个用逗号隔开的数字,求出这两个数字的和,并且用正常的方式输出来.直接写一个函数将一个包含逗号的数字转换成十进制的数返回就行了.这里推荐一个函数atoi(),参数是char*型的,然 ...
- F - A计划
题目链接: https://cn.vjudge.net/contest/254150#problem/F wa代码: #include<iostream> #include<stri ...
- redhat7配置本地yum源
1.首先是要有一个iso文件,并将这个文件挂载到某个目录 挂载: 配置: 检验: yum list 现在你就可以在没有网的情况下,安装软件了~~~
- mysql集成部署
经常听说mysql数据库是集成在系统中,也一直不太明白集成的概念.今天才明白集成的概念就是将mysql所有的文件放到一个文件夹下放到系统中,也就是将mysql采用目录迁移部署的方式进行安装.在上一篇研 ...
- grep和sed匹配多个字符关键字的用法
GNU sed和UNIX sed 写法不一样 匹配多个关键词,打印出匹配的行,效果类似于 grep grep hello\|world file > output 或者用扩展正则 grep -E ...
- 011_自定义mac通知的时长
打开终端(找不到的点击 Mac 屏幕右上角放大镜按钮,Spotlight 搜索 “终端”),粘入下面这行命令,回车就行了.注意最后的 # 替换成你希望通知中心横幅停留的秒数,比如 15.default ...