Java多线程之内存可见性

标签: java 多线程 内存 | 发表时间:2015-05-29 03:47 | 作者:u012422829
出处:http://blog.csdn.net

一、JAVA内存模型简介

JAVA Merory  Model描述了JAVA程序中各种变量(线程共享变量)的访问规则,以及在JVM中将变量存储到内存和从内存中读取变量这样的底层细节。

所有的变量都保存在主内存中,但是每个线程都有自己的独立工作内存,保存该线程使用到的变量的一个副本。


两条规定

1.线程对共享变量的操作只能在独立的工作内存中进行,不能在主内存中直接读写;

2.不同线程之间无法直接访问其他线程的工作内存中的变量,需要通过主内存完成。

实现共享变量的可见性,必须保证两点

1.线程修改后的共享变量能及时从工作内存刷新到主内存中;

2.其他线程能及时把共享变量的最新值从主内存中更新到工作内存中。

JAVA从语言层面支持的可见性实现方式

synchronized; volatile


二、synchronized实现可见性


synchronized除了使代码同时只能被一个线程执行的功能,还有一个功能就是实现可见性。

关于synchronized的两条规定

1.线程解锁前,必须把共享变量的最新值刷到主内存去;
2.线程加锁时,清空工作内存的共享变量值,从主内存中重新读取。 
这两条规定保证了线程在解锁前对共享变量的操作在下次加锁前对其他线程可见。

synchronized实现可见性的过程


1.获得互斥锁
2.清空工作内存
3.从主内存中拷贝共享变量的最新值到工作内存
4.执行代码
5.把修改后的变量值更新到主内存
6.释放锁

as-if-serial与重排序

重排序: 代码的执行顺序和书写顺序不相同,指令重排序是编译器或处理器为了提高性能而做的优化。
分为 编译器优化、处理器指令级并行重排序、内存系统的重排序(对读写缓存的优化,如上所说的那些)


as-if-serial:
无论如何重排序,程序执行的结果与代码顺序执行的结果相一致。(这并不是说就不进行重排序了!)

Java在单线程下是保证as-if-serial的。


导致共享变量在线程间不可见的原因及解决方案

1.线程交叉执行;  synchronized的原子性解决这个问题
2.重排序+线程交叉执行(甚至if() 与 {}中都可能被重排序,控制依赖关系并不是重排序的约束,只有数据依赖关系才是) synchronized的原子性解决这个问题
3.共享变量更新后的值没有在主内存与工作内存之间及时更新 synchronized的两条规定解决这个问题


三、volatile实现可见性


volatile关键字可以保证volatile变量的可见性,但是不能保证volatile变量复合操作的原子性。

volatile如何实现

通过加入内存屏障和禁止指令重排序来实现:

对volatile变量执行写操作时,会在写操作后加一条store屏障指令,会把CPU写缓存的缓存强制刷新到主内存,还能防止处理器把volatile前的变量重排序到volatile之后。
对volatile变量执行读操作时,会在读操作前加一条load屏障指令。在读取时强制从主内存中读取。

通俗的说,就是volatile变量每次被线程访问时,都强制从主内存中重读该变量的值,而不是从工作内存的缓存中;当该值发生变化时,强制线程把最新的值刷新到主内存中。

不能保证volatile变量复合操作的原子性

比如 :
private  volatile int a=0;
a++;   //不是原子操作:分三步, 读取a的值,a+1;写回a的值

但是下面的是原子操作:
synchronized(this){
a++;
}

除了synchronized之外,还可以用Lock:
Lock lock= new ReentrantLock();  //可重入锁
然后在后面:
lock.lock();
try{
a++;
}finally{
lock.unlock();
}

volatile的适用场景

1.对变量的写入操作不依赖当前值
   不满足: num++; num=num*5 ...
   满足:    布尔变量等等
2.该变量没有包含在具有其他变量的不变式中
   即程序中有多个volatile变量时,各自要独立。如a,b;  如果有a<b这样的式子,就不满足。

所以volatie的适用范围不如synchronized大。

四、synchronized与volatile的比较


1.volatile不需要加锁,比synchronized轻量级,不会造成线程阻塞。
2.synchronized既可以保证可见性,又可以保证原子性;而volatile只能保证可见性。



===========================================================================
(文章是看完慕课的视频后写的,感谢 http://www.imooc.com/view/352)

转载请注明出处,谢谢!

作者:u012422829 发表于2015/5/28 19:47:10 原文链接
阅读:43 评论:0 查看评论

相关 [java 多线程 内存] 推荐:

Java 多线程内存模型

- - ITeye博客
Java 多线程内存模型.       Java虚拟机规范中试图定义一种Java内存模型(Java Memory Model,JMM)来屏蔽掉各种硬件和操作系统的内存访问差异,以实现让Java程序在各种平台下都能达到一致的并发效果. 在此之前,主流程序怨言(如C/C++等)直接使用物理硬件(或者说操作系统的内存模型),因此,会由于不同的平台上内存模型差异,导致程序在一套平台上并发完成正常,而在另一套平台上并发访问却经常出错,因此经常需要针对不同的平台来编写程序.

Java多线程之内存可见性

- - CSDN博客推荐文章
一、JAVA内存模型简介. JAVA Merory  Model描述了JAVA程序中各种变量(线程共享变量)的访问规则,以及在JVM中将变量存储到内存和从内存中读取变量这样的底层细节. 所有的变量都保存在主内存中,但是每个线程都有自己的独立工作内存,保存该线程使用到的变量的一个副本. 1.线程对共享变量的操作只能在独立的工作内存中进行,不能在主内存中直接读写;.

Java Thread多线程

- - CSDN博客推荐文章
Java Thread多线程. Java 多线程例子1 小例子. super("zhuyong");//设置线程的名字,默认为“TestThread”. Java 多线程例子2 前台线程(用户线程) 后台线程(守护线程 ). 1,setDaemon(true)后就是后台线程(守护线程 ),反之就是前台线程(用户线程).

Java多线程之synchronized

- - CSDN博客推荐文章
这里通过三个测试类阐述了synchronized应用的不同场景. 首先是最基本的synchronized Method的使用.  * @see 概述:Java中的每个对象都有一个锁(lock)或者叫做监视器(monitor) .  * @see 说明:当synchronized关键字修饰一个方法时,则该方法为同步方法 .

java多线程总结

- - Java - 编程语言 - ITeye博客
在java中要想实现多线程,有两种手段,一种是继续Thread类,另外一种是实现Runable接口. 对于直接继承Thread的类来说,代码大致框架是:. class 类名 extends Thread{. * @author Rollen-Holt 继承Thread类,直接调用run方法.             System.out.println(name + "运行     " + i);.

Java多线程学习

- - CSDN博客编程语言推荐文章
  线程是一种轻量级的进程,它和进程一样拥有独立的执行控制,由操作系统负责调度,区别在于线程没有独立的存储空间,而是和所属进程中的其它线程共享一个存储空间,这使得线程间的通信远较进程简单. 即多个线程可以同时执行,就像有多条流水线一样,可以同时进行工作,是并发执行的.   程序是由进程组成的,进程是由线程组成的.

Java多线程(二)同步

- - CSDN博客编程语言推荐文章
如果你的java基础较弱,或者不大了解java多线程请先看这篇文章 java多线程(一)线程定义、状态和属性. 同步一直是java多线程的难点,在我们做android开发时也很少应用,但这并不是我们不熟悉同步的理由. 希望这篇文章能使更多的人能够了解并且应用java的同步. 在多线程的应用中,两个或者两个以上的线程需要共享对同一个数据的存取.

[Java] Java 多线程案例分析

- - V2EX
现要从 hbase中导出 2016 年整年的,大约 10w只股票行情数据,数据总量约 100t. 汇总到 hdfs中供需求方使用. 已知数据量规模大概是 100t,那么单台机器处理肯定不是不行的,先不说大多数磁盘都没这么大,即便磁盘有这么大,单台机器处理对于内存和 cpu 要求也很高,所以我们将问题一般化,使用数量有限的低配机器.

Java多线程之wait()和notify()

- - CSDN博客推荐文章
直接看测试代码吧,细节之处,详见注释.  * Java多线程之wait()和notify()的妙用 .  * @see 问题:同时启动两个线程和同时启动四个线程,控制台打印结果是不同的 .  * @see      同时启动两个线程时,控制台会很规律的输出1010101010101010 .  * @see      同时启动四个线程时,控制台起初会规律的输出10101010,一旦某一刻输出一个负数,那么后面的输出就会"一错再错" .

java多线程设计wait/notify机制

- - CSDN博客推荐文章
  当线程A获得了obj锁后,发现条件condition不满足,无法继续下一处理,于是线程A就wait() , 放弃对象锁..   之后在另一线程B中,如果B更改了某些条件,使得线程A的condition条件满足了,就可以唤醒线程A:.   # 调用obj的wait(), notify()方法前,必须获得obj锁,也就是必须写在synchronized(obj) {…} 代码段内.