【慕课网学习笔记】Java共享变量的可见性和原子性
1. Java内存模型(Java Memory Model, JMM)
Java的内存模型如下,所有变量都存储在主内存中,每个线程都有自己的工作内存。
共享变量:如果一个变量在多个线程中都使用到了,那么这个变量就是这几个线程的共享变量。
可见性:一个线程对共享变量的修改,能够及时地到主内存并且让其他的线程看到。
怎么理解上面的可见性的意思呢?
线程对共享变量的修改,只能在自己的工作内存里操作,不能直接对主内存中的共享变量进行修改。而且一个线程不能直接访问另一个线程中的变量的值,只能通过主内存进行共享传递。
那么就要求线程A对共享变量修改后,及时地更新到主内存中,线程B才可以及时地从主内存获取最新的值到工作内存。
比如一个共享变量int i = 0; 线程A将其改为i =1; 其他线程此时获取i的值,应该能及时地得到1,而不是0。

2. synchronized实现可见性
synchronized除了常见的原子性,还实现了可见性。这是因为:
1) 线程解锁前,必须把共享变量的最新值刷新到主内存中去;
2) 线程加锁时,将清空工作内存中的共享变量的值,使用到共享变量时,从主内存中获取最新的共享变量值(加锁和解锁需要同一把锁)
3. volatile实现可见性
通过内存屏障和禁止重排序优化来实现可见性。
1) 对共享变量进行写操作后,加入一条store屏障指令,强制将共享变量的值刷新到主内存;
2) 对共享变量进行读操作前,加入一条load屏障指令,强制从主内存中将最新值刷新到工作内存;
4.volatile不能保证原子性
一个比较典型的例子是++运算符。
在下面的代码中,一共创建了1000个线程,预期应该是加了1000次,那么number的值应该是1000,实际上有可能并不是。
这是因为,++运算符并不是一次操作。以number++为例,可以看作是,先从主内存中取出number的值,然后将其加1,刷新工作内存,刷新主内存,这么几个步骤。
而volatile并不能保证原子性,这就意味着,有可能出现这种情况:
1)线程A获取到主内存的number的值(假设为10)到工作内存
2)此时CPU调度,A暂停,线程B开始执行,同样从主内存中获取到number为10,number++后,number为11,刷新到主内存
3)线程A继续执行number++,它的工作内存中number为10,执行完毕刷新到主内存,此时,number的值为11. 也就是说,AB两个线程同时进行了+1操作,但最终的结果,只加了1
public class VolatileDemo {
private int number = 0;
public void increase() {
number++;
}
public int getNumber() {
return number;
}
public static void main(String[] args) {
final VolatileDemo demo = new VolatileDemo();
for (int i = 0; i <= 999; i++) {
new Thread(new Runnable() {
@Override
public void run() {
demo.increase();
}
}).start();
}
//线程未执行完,主线程让出CPU资源
while(Thread.activeCount() > 1){
Thread.yield();
}
//待上面的线程都执行完了,再打印,避免打印的不是最后的数据
System.out.println(demo.getNumber());
}
}
5.volatile适用场景
1)对共享变量的写操作,不依赖于其之前的值
不合适:number++, number = number * 2, number += 1等
合适:boolean值
2)该变量没有包含在具有其他变量的不变式中,也就是说,不同的volatile变量之间,不能互相依赖
6.AtomicInteger实现递增
上面我们已经知道一个整型的共享变量要实现递增,如果使用++运算符,即使加上volatile关键字,也是无法保证其原子性的。而如果在访问变量时加上synchronized块,或者可重入锁,开销又太大。
JDK1.5之后,可以使用AtomicInteger进行递增。该类是线程安全的。
将上面的代码修改如下,就可以保证原子性和可见性。
import java.util.concurrent.atomic.AtomicInteger;
public class VolatileDemo {
private AtomicInteger number = new AtomicInteger(0);
public void increase() {
number.incrementAndGet();
}
public int getNumber() {
return number.intValue();
}
public static void main(String[] args) {
final VolatileDemo demo = new VolatileDemo();
for (int i = 0; i <= 999; i++) {
new Thread(new Runnable() {
@Override
public void run() {
demo.increase();
}
}).start();
}
//线程未执行完,主线程让出CPU资源
while(Thread.activeCount() > 1){
Thread.yield();
}
//待上面的线程都执行完了,再打印,避免打印的不是最后的数据
System.out.println(demo.getNumber());
}
}
【慕课网学习笔记】Java共享变量的可见性和原子性的更多相关文章
- JavaScript进阶--慕课网学习笔记
JAVASCRIPT—进阶篇 给变量取个名字(变量命名) 变量名字可以任意取,只不过取名字要遵循一些规则: 1.必须以字母.下划线或美元符号开头,后面可以跟字 ...
- JavaScript入门--慕课网学习笔记
JAVASCRIPT—(慕课网)入门篇 我们来看看如何写入JS代码?你只需一步操作,使用<script>标签在HTML网页中插入JavaScript代码.注意, <script&g ...
- HTML基本语法(慕课网学习笔记)
标题 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8& ...
- 初识javaScript(慕课网学习笔记)
js输出 window.alert() 警告框 document.write() 写到HTML文档中 innerHTML 写到HTML元素 console.log() 写到浏览器的控制台 <!D ...
- Float浮动(慕课网学习笔记)
float浮动 属性:值 意义 float:left 左浮动 float:right 右浮动 float:none 不浮动 float:inherit 继承父元素浮动属性,若父元素没有浮动属性则失效 ...
- CSS基本语法(慕课网学习笔记)
CSS的声明,内外联样式以及CSS的优先级 css学习.html <!DOCTYPE html> <html lang="en"> <head> ...
- js之window对象(慕课网学习笔记)
javaScript定义了一个变量一个函数都会变成window中的一个成员 var a=1; alert(window.a) //会输出a的值 window基础 创建窗口.调整窗口.移动窗口.关闭窗口 ...
- js之DOM入门(慕课网学习笔记)
DOM简介 获得元素 document.getElementById('') 1.通过id获得元素内容 document.getElementsByTagName('') 2.通过标签获得元素内容 d ...
- CSS定位(慕课网学习笔记)
定位模型 static自然模型 relative相对定位模型 absolute绝对定位模型 fixed固定定位模型 sticky磁铁定位模型 possition之static(默认的设置)(静态定位. ...
随机推荐
- ruby编程语言-学习笔记3(第4章 表达式和操作符)
4.6 操作符 了解优先级很重要 位移操作符 (0b1011)<< 1 # ==> "10110" 11 << 1 = 22 ...
- C#使用SQLite出错:无法加载 DLL“SQLite.Interop.dll”,找不到指定的模块
在SQLite官方下载了System.Data.SQLite,编写如下测试代码: 复制内容到剪贴板 程序代码 using (SQLiteConnection conn = new SQLiteConn ...
- HDU5654xiaoxin and his watermelon candy 离线+树状数组
题意:bc 77div1 d题(中文题面),其实就是询问一个区间有多少不同的三元组,当然这个三元组要符合条件 分析(先奉上官方题解) 首先将数列中所有满足条件的三元组处理出来,数量不会超过 nn个. ...
- lua package.path的使用
需要用lua写一个工具,c++调用lua,然后这个lua要require其他lua脚本,在主lua里面设置package.path,但一直都失败,甚至lua都无法编译通过. project_path ...
- mysql server install
1.首先在mysql的官网www.mysql.com或者其他网站下载mysql.zip或者mis格式的文件目前5.6的差不多300多M. 2.zip压缩包是绿色版的不用安装,直接用dos命令操作就行. ...
- git引用^和~的区别
这篇git文章必转:解答我一直的疑惑 http://www.cnblogs.com/hutaoer/archive/2013/05/14/3078191.html 一. 引子 在git操作中,我们可以 ...
- normalization归一化
简单的举个例子:一张表有两个变量,一个是体重kg,一个是身高cm.假设一般情况下体重这个变量均值为60(kg),身高均值为170(cm).1,这两个变量对应的单位不一样,同样是100,对于身高来说很矮 ...
- Linux下platform设备以及platform设备和驱动注册的先后顺序
platform是Linux系统提供的一种管理设备的手段,所有SOC系统中集成的独立的外设控制器.挂接在SOC内存空间的外设等都属Platform设备.如ARM S3C6410处理器中,把内部集成的I ...
- 【解决】Oracle服务器ip地址被占用
数据库服务器ip地址被占用,怎么破?! 服务器: 1.改服务器ip: 2.改tnsnames.ora里配置的Oracle数据库ip: 3.重启Oracle服务: 客户端: 1.改tnsnames.or ...
- linux统计文件夹某一些文件的大小总和
du -m smallgame_2006* | awk '{sum += $1}; END{print sum}' -m代表单位是MB, awk命令需要'',且命令需换行