一、目录

  1、引入话题-发散思考

  2、volatile深度解析

  3、解决volatile原子性问题

  4、volatile应用场景

二、引入话题-发散思考

  1. public class T1 {
  2. /*volatile*/ boolean running=true;
  3.  </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> m(){
  4.       System.out.println(Thread.currentThread().getName()</span>+":start!"<span style="color: #000000;">);
  5.        </span><span style="color: #0000ff;">while</span><span style="color: #000000;">(running){
  6.             </span><span style="color: #008000;">/*</span><span style="color: #008000;">try {
  7.                  TimeUnit.MINUTES.sleep(2);
  8.             } catch (Exception e) {
  9.                  e.printStackTrace();
  10.             }</span><span style="color: #008000;">*/</span><span style="color: #000000;">
  11.        }
  12.       System.out.println(Thread.currentThread().getName()</span>+":end!"<span style="color: #000000;">);
  13.  }
  14.  </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> main(String[] args) {
  15.        T1 t</span>=<span style="color: #0000ff;">new</span><span style="color: #000000;"> T1();
  16.        </span><span style="color: #0000ff;">new</span> Thread(()-&gt;t.m(),"t"<span style="color: #000000;">).start();
  17.        </span><span style="color: #0000ff;">try</span><span style="color: #000000;"> {
  18.             TimeUnit.SECONDS.sleep(</span>1<span style="color: #000000;">);
  19.        } </span><span style="color: #0000ff;">catch</span><span style="color: #000000;"> (Exception e) {
  20.             e.printStackTrace();
  21.        }
  22.        t.running</span>=<span style="color: #0000ff;">false</span><span style="color: #000000;">;
  23.  }
  24.  
  25. }

  26. 运行结果:

  27. volatile

  28. t:start!

  29. volatile:

  30. t:start!

  31. t:end!

含有volatile是期望的结果,那为什么不添加volatile会产生这种情况呢?
再谈Java内存模型:
     在虚拟机中,堆内存用于存储共享数据(实例对象),堆内存也就是这里说的主内存。
     每个线程将会在堆内存中开辟一块空间叫做线程的工作内存,附带一块缓存区用于存储共享数据副本。那么,共享数据在堆内存当中,线程通信就是通过主内存为中介,线程在本地内存读并且操作完共享变量操作完毕以后,把值写入主内存。
 
解析无volatile:
  • 根据上述java内存模型可知,最开始running=true在主存中,开启线程A,线程会把主存的running=true复制一份写入工作内存的共享变量副本中。
  • 当我们改变running=false,在主存中已经发生改变。
  • 线程A一直在工作状态,没有空闲时间去知道主存的情况,而是一直在读本地内存的共享变量副本,也就一直running=true,取而代之也会产生上述情况。

三、volatile深度解析

那为什么含有volatile就能及时刷新工作内存呢?它有什么作用呢?
volatile:可见性(一个线程修改共享变量以后,立马会被其他线程知可见)、禁止重排序。
 
1、什么是可见性?
虚拟机的happens-before中的volatile规则:volatile变量写操作先于读操作,一个线程去读取volatile变量,另一个线程去写volatile变量,那么volatile变量的写操作优先。
  • 根据上述java内存模型可知,最开始running=true在主存中,开启线程A,线程会把主存的running=true复制一份写入工作内存的共享变量副本中。
  • 当我们改变running=false,在主存已经发生改变。
  • 就在这时,当主存与工作内存发生不一致的时候,工作内存的共享变量会失效,那么工作内存就会去主存刷新一遍共享变量,所以running=false,自然就执行下面的代码啦!
2、什么是禁止重排序?
先谈有序性:
  1. int a=1;
  2. int b =3;
  3. int c=a*b
在虚拟机中,执行上述代码,一定是按照上述顺序执行吗?那可不一定,像a=1,b=3的顺序完全可能先执行b=3,a=1,这被称为重排序;但是c=a*b一定在a=1,b=3后面,这被成为有序性。
 
再谈重排序:
  1. //线程1:
  2. context = loadContext(); //语句1
  3. inited = true; //语句2
  4.  
  5. //线程2:

  6. while(!inited ){

  7. sleep()

  8. }

  9. doSomethingwithconfig(context);

假设上述代码在单线程中,谈过重排序不会对代码造成什么影响,但是我们看这一段代码。
语句1与语句2并没有太多的依赖关系,参考有序性例子,那么他们就可以重排序,那么可能语句2执行先于语句1,假设在线程1中语句2执行完就刷新inited到主存,还没等语句1执行呢?线程2就执行起来,一看inited=true,挑出循环,执行下面的代码,context=null就报出空指针,显然这是不能被虚拟机允许的。
 
所以,volatile明确规定禁止重排序,意思就是context=loadContext必须先于inited执行。
在虚拟机中设定了有序性,也就是前面谈到的happens-before原则。如果不满足上该原则的情况下,虚拟机是可以自由的重排序的,下面附录此规则。
 
3、先行发生原则(happens-before)
    • 程序次序规则:一个线程内,按照代码顺序,书写在前面的操作先行发生于书写在后面的操作
    • 锁定规则:一个unLock操作先行发生于后面对同一个锁的lock操作
    • volatile变量规则:对一个变量的写操作先行发生于后面对这个变量的读操作
    • 传递规则:如果操作A先行发生于操作B,而操作B又先行发生于操作C,则可以得出操作A先行发生于操作C
    • 线程启动规则:Thread对象的start()方法先行发生于此线程的每个一个动作
    • 线程中断规则:对线程interrupt()方法的调用先行发生于被中断线程的代码检测到中断事件的发生
    • 线程终结规则:线程中所有的操作都先行发生于线程的终止检测,我们可以通过Thread.join()方法结束、Thread.isAlive()的返回值手段检测到线程已经终止执行
    • 对象终结规则:一个对象的初始化完成先行发生于他的finalize()方法的开始

四、解决volatile原子性问题

1、volatile能解决原子性问题吗?什么是原子性呢,本不想解释,为了读者能够更透彻理解,再解释一下。

原子性:只有一个线程访问共享数据,也就是当线程A访问一个代码块的时候,其他线程全部堵塞,只有等代码块全部执行完,才能被其他线程访问共享数据。
  1. public class T2 {
  2. volatile int count=0;
  3.  </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> m(){
  4.        </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=0;i&lt;1000;i++<span style="color: #000000;">)
  5.             count</span>++<span style="color: #000000;">;
  6.  }
  7.  </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> main(String[] args) {
  8.        T2 t</span>=<span style="color: #0000ff;">new</span><span style="color: #000000;"> T2();
  9.        List</span>&lt;Thread&gt; threads=<span style="color: #0000ff;">new</span> ArrayList&lt;Thread&gt;<span style="color: #000000;">();
  10.        </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=0;i&lt;10;i++<span style="color: #000000;">){
  11.             threads.add(</span><span style="color: #0000ff;">new</span> Thread(()-&gt;t.m(),"thread-"+<span style="color: #000000;">i));
  12.        }
  13.        threads.forEach((o)</span>-&gt;<span style="color: #000000;">o.start());
  14.        </span><span style="color: #008000;">//</span><span style="color: #008000;">等待所有线程都执行完</span>
  15.        threads.forEach((o)-&gt;<span style="color: #000000;">o.yield());
  16.        System.out.println(</span>"count:"+<span style="color: #000000;">t.count);
  17.  }
  18.  
  19. }

  20. 运行结果:

  21. count8710 //每次都不一样。

 2、为什么加了volatile还是不能得到预期结果呢?因为它只保证了可见性,不能保证原子性。what?

再回忆java内存模型:

 

3、那怎么解决呢?

方式一:synchronized,jvm对synchronized进行了很大的优化,所以效率也没有想象中那么低。

  1. public class T3 {
  2. int count=0;
  3.  </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">synchronized</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> m(){
  4.        </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=0;i&lt;1000;i++<span style="color: #000000;">)
  5.             count</span>++<span style="color: #000000;">;
  6.  }
  7.  </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> main(String[] args) {
  8.        T3 t</span>=<span style="color: #0000ff;">new</span><span style="color: #000000;"> T3();
  9.        List</span>&lt;Thread&gt; threads=<span style="color: #0000ff;">new</span> ArrayList&lt;Thread&gt;<span style="color: #000000;">();
  10.        </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=0;i&lt;10;i++<span style="color: #000000;">){
  11.             threads.add(</span><span style="color: #0000ff;">new</span> Thread(()-&gt;t.m(),"thread-"+<span style="color: #000000;">i));
  12.        }
  13.        threads.forEach((o)</span>-&gt;<span style="color: #000000;">o.start());
  14.        </span><span style="color: #008000;">//</span><span style="color: #008000;">等待所有线程都执行完</span>
  15.        threads.forEach((o)-&gt;<span style="color: #000000;">o.yield());
  16.        System.out.println(</span>"count:"+<span style="color: #000000;">t.count);
  17.  }
  18.  
  19. }

方式二:ReentrantLock,跟synchronized的作用差不多。

  1. public class T5 {
  2. ReentrantLock lock=new ReentrantLock();
  3. int count=0;
  4.  </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> m(){
  5.        lock.lock();
  6.        </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=0;i&lt;1000;i++<span style="color: #000000;">)
  7.             count</span>++<span style="color: #000000;">;
  8.        lock.unlock();
  9.  }
  10.  </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> main(String[] args) {
  11.        T4 t</span>=<span style="color: #0000ff;">new</span><span style="color: #000000;"> T4();
  12.        List</span>&lt;Thread&gt; threads=<span style="color: #0000ff;">new</span> ArrayList&lt;Thread&gt;<span style="color: #000000;">();
  13.        </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=0;i&lt;10;i++<span style="color: #000000;">){
  14.             threads.add(</span><span style="color: #0000ff;">new</span> Thread(()-&gt;t.m(),"thread-"+<span style="color: #000000;">i));
  15.        }
  16.        threads.forEach((o)</span>-&gt;<span style="color: #000000;">o.start());
  17.        </span><span style="color: #008000;">//</span><span style="color: #008000;">等待所有线程都执行完</span>
  18.        threads.forEach((o)-&gt;<span style="color: #000000;">o.yield());
  19.        System.out.println(</span>"count:"+<span style="color: #000000;">t.count);
  20.  }
  21.  
  22. }

方式三:AtomicInteger原子类
  1. public class T4 {
  2. AtomicInteger count=new AtomicInteger(0);
  3.  </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> m(){
  4.        </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=0;i&lt;1000;i++<span style="color: #000000;">)
  5.             count.getAndIncrement();
  6.  }
  7.  </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> main(String[] args) {
  8.        T4 t</span>=<span style="color: #0000ff;">new</span><span style="color: #000000;"> T4();
  9.        List</span>&lt;Thread&gt; threads=<span style="color: #0000ff;">new</span> ArrayList&lt;Thread&gt;<span style="color: #000000;">();
  10.        </span><span style="color: #0000ff;">for</span>(<span style="color: #0000ff;">int</span> i=0;i&lt;10;i++<span style="color: #000000;">){
  11.             threads.add(</span><span style="color: #0000ff;">new</span> Thread(()-&gt;t.m(),"thread-"+<span style="color: #000000;">i));
  12.        }
  13.        threads.forEach((o)</span>-&gt;<span style="color: #000000;">o.start());
  14.        </span><span style="color: #008000;">//</span><span style="color: #008000;">等待所有线程都执行完</span>
  15.        threads.forEach((o)-&gt;<span style="color: #000000;">o.yield());
  16.        System.out.println(</span>"count:"+<span style="color: #000000;">t.count);
  17.  }
  18.  
  19. }

五、volatile应用场景

说到这里,读者可能就已经懵逼了,这也有问题那也有问题,那我们什么时候用它呢?
volatile是基于synchronized提出的效率优化手段,但是它是不能代替synchronized的。
 
状态标记量:
一般volatile共享变量,不要用于数据计算,最好去标记一些状态值,比如前面说的running=true。
  1. volatile boolean inited = false;
  2. //线程1:
  3. context = loadContext();
  4. inited = true;
  5.  
  6. //线程2:

  7. while(!inited ){

  8. sleep()

  9. }

  10. doSomethingwithconfig(context);

九、版权声明

  作者:邱勇Aaron

  出处:http://www.cnblogs.com/qiuyong/

  您的支持是对博主深入思考总结的最大鼓励。

  本文版权归作者所有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,尊重作者的劳动成果。

  参考:深入理解JVM、马士兵并发编程、并发编程实践

     volatile关键字解析:http://www.importnew.com/18126.html

并发编程(三):全视角解析volatile的更多相关文章

  1. 并发编程(二):全视角解析volatile

    一.目录 1.引入话题-发散思考 2.volatile深度解析 3.解决volatile原子性问题 4.volatile应用场景 二.引入话题-发散思考 public class T1 { /*vol ...

  2. [Java并发编程(五)] Java volatile 的实现原理

    [Java并发编程(五)] Java volatile 的实现原理 简介 在多线程并发编程中 synchronized 和 volatile 都扮演着重要的角色,volatile 是轻量级的 sync ...

  3. [Java并发编程(四)] Java volatile 的理论实践

    [Java并发编程(四)] Java volatile 的理论实践 摘要 Java 语言中的 volatile 变量可以被看作是一种 "程度较轻的 synchronized":与 ...

  4. Java并发编程三个性质:原子性、可见性、有序性

      并发编程 并发程序要正确地执行,必须要保证其具备原子性.可见性以及有序性:只要有一个没有被保证,就有可能会导致程序运行不正确  线程不安全在编译.测试甚至上线使用时,并不一定能发现,因为受到当时的 ...

  5. 干货:Java并发编程必懂知识点解析

    本文大纲 并发编程三要素 原子性 原子,即一个不可再被分割的颗粒.在Java中原子性指的是一个或多个操作要么全部执行成功要么全部执行失败. 有序性 程序执行的顺序按照代码的先后顺序执行.(处理器可能会 ...

  6. Java并发编程中的设计模式解析(二)一个单例的七种写法

    Java单例模式是最常见的设计模式之一,广泛应用于各种框架.中间件和应用开发中.单例模式实现起来比较简单,基本是每个Java工程师都能信手拈来的,本文将结合多线程.类的加载等知识,系统地介绍一下单例模 ...

  7. 从缓存入门到并发编程三要素详解 Java中 volatile 、final 等关键字解析案例

    引入高速缓存概念 在计算机在执行程序时,以指令为单位来执行,每条指令都是在CPU中执行的,而执行指令过程中,势必涉及到数据的读取和写入. 由于程序运行过程中的临时数据是存放在主存(物理内存)当中的,这 ...

  8. JAVA并发编程:相关概念及VOLATILE关键字解析

    一.内存模型的相关概念 由于计算机在执行程序时都是在CPU中运行,临时数据存在主存即物理内存,数据的读取和写入都要和内存交互,CPU的运行速度远远快于内存,会大大降低程序执行的速度,于是就有了高速缓存 ...

  9. Java并发编程、内存模型与Volatile

    http://www.importnew.com/24082.html  volatile关键字 http://www.importnew.com/16142.html  ConcurrentHash ...

随机推荐

  1. python中的一些小知识

    在最近学习python中遇到的一些小问题汇总一下: 1.在windows7下安装python3.5版本时提示安装不了,缺少ServicePack1.  解决办法是,打开控制面板\系统和安全\Windo ...

  2. Spring框架入门

    技术分析之什么是Spring框架        1. Spring框架的概述        * Spring是一个开源框架        * Spring是于2003 年兴起的一个轻量级的Java开发 ...

  3. 关于爬楼梯的lintcode代码

    讲真的,这个我只会用递归去做,但是lintcode上面超时,所以只有在网上找了个动态规划的,虽然这个程序懂了,但是我觉得还是挺不容易的真正弄懂的话-- class Solution {public:  ...

  4. Fast Walsh-Hadamard Transform——快速沃尔什变换

    模板题: 给定$n = 2^k$和两个序列$A_{0..n-1}$, $B_{0..n-1}$,求 $$C_i = \sum_{j \oplus k = i} A_j B_k$$ 其中$\oplus$ ...

  5. MapReduce运行流程分析

    研究MapReduce已经有一段时间了.起初是从分析WordCount程序开始,后来开始阅读Hadoop源码,自认为已经看清MapReduce的运行流程.现在把自己的理解贴出来,与大家分享,欢迎纠错. ...

  6. Hibernate createQuery调用joincolumn

    1. Beans a. Version Bean package locationService.beans; import java.sql.Timestamp; import java.util. ...

  7. STM32的RFID射频读写控制装置

    ,大二上学期做的,过了很久,先上一下图: 这并不是做个最后一版:主体是RC552+STM32+1062:蜂鸣器,继电器,LED等:反正最后的效果就是,刷一下卡,1602显示一下持卡人(需要提前注册,注 ...

  8. NFS文件共享

    NFS文件共享 简介 NFS即网络文件系统(network file system),监听在TCP 2049端口. 服务器需要记住客户端的ip地址以及相应的端口信息,这些信息可以委托给RPC(remo ...

  9. C/C++遍历目录下的所有文件(Windows/Linux篇,超详细)

    本文可转载,转载请注明出处:http://www.cnblogs.com/collectionne/p/6815924.html. 前面的一篇文章我们讲了用Windows API遍历一个目录下的所有文 ...

  10. 关于python编译的一点小结

    大家都知道python是脚本语言,源码可以直接执行,有时需要提高执行效率或者保密(因为有时候不想让使用人看到源码文件),那就涉及到python编译了,那么该如何做呢? 有两种方法可以做到. 1.一种是 ...