美文网首页
volatile关键字

volatile关键字

作者: JingJingLiu317 | 来源:发表于2018-04-19 21:32 被阅读0次

    volatile是Java中的关键字,是轻量级的并发实现,效率比synchronized高,唯一不足是不能保证原子性,可保证有序性和内存可见性。

    本节内容如下:

    1.讲解Java内存模型

    2.并发的三大特性:原子性,有序性,可见性

    3.深入理解volitale

    4. volatile和synchronized区别:

    1.Java内存模型

    Java内存模型规定,所有的共享变量都存储在主内存中,每个线程还有自己的工作内存,线程工作时,将主存中的变量拷贝到自己的工作内存,线程的读取、赋值等操作都是在工作内存中完成的,修改后的值不确定什么时候会同步到主存中,线程之间不能互相访问对方工作内存,这样线程工作内存中的值并不是最新值,从而产生内存不一致问题。

    如:i=++i;

    若初始化i=0,两个线程同时访问,最终i应该为2。但是可能会出现下面这种情况,两个线程同时读取到工作内存的值均为0,第一个线程执行完成后将1写到主内存,第二个线程的工作内存中i还为0,计算完成后,写到主存的值仍然为1。对于数字的操作,Java提供了原子操作类:AtomicInteger.

    2. 并发三大特性

    2.1 原子性

    原子性即一个操作或多个操作,要么全部执行成功,要么全部执行失败。

    比如:A给B赚钱,A账户减钱和B账户加钱的操作就是原子性的,必须要同时成功或同时失败,不可能A减钱成功,B加钱失败。

    Java中的读取和赋值的单个操作都是原子性的,但是组合起来的操作就不是原子性的了。如下:

    x=10;将10写到工作内存,是原子性的

    y=x;该赋值包含两步,读取x的值,然后将10写到工作内存,所以并非原子性

    x++;该操作包含三步,读取x的值,将x的值加1,将运行后的值写到工作内存中,所以也并非原子性

    2.2 有序性

    有序性是指程序执行的顺序按照代码的顺序执行。

    而程序真正执行时不一定是按照代码的顺序执行的,因为可能会发生指令重排序,在指令重排序时会考虑数据间的依赖性,被依赖的数据会先执行,保证执行结果正确。适当的指令重排序会提升程序的性能。

    虽然重排序不会影响单线程的执行结果,但是对于多线程的执行可能会导致结果错误。

    如下:

    //线程1执行如下代码

    context=loadContext();//1

    flag=true;//2

    //线程2执行如下代码

    while(!flag){}

    dosomething(context);

    1和2重排序后,单线程环境下不会影响结果,多线程时,若线程1执行完2还没初始化context,线程2跳出循环执行下面一条语句则会报错。

    Java中可以通过volitale关键字保证有序性,防止编译器和处理器对指令进行重排序。

    2.3 内存可见性

    内存可见性是指,当多个线程共享同一个变量时,一个线程修改了该变量,其他线程能立刻看到修改后的新值。

    当主线程将flag改为false后,由于没有立即写到主内存,第一个线程一直在工作内存读取flag,则很可能会造成死循环或者很久才读到最新变量。

    3.深入理解volitale

    volitale是线程同步的轻量级实现。一个共享变量被volitale修饰之后,即可保证内存可见性和有序性,但是不能保证原子性。

    3.1 内存可见性

    被volitale修饰后,线程改变共享变量后,会强制刷新到主内存,使其他线程的工作内存中该变量的缓存失效,所以其他线程再次读取该变量时只能去主存中读。

    3.2 有序性

    volitale可以通过禁止指令重排序来实现有序性。

    如何做到禁止指令重排序的呢?

    如下例子:

    x=0;//1

    y=1;//2

    volatile boolean flag=true;//3

    x=x+1;//4

    y=y+1;//5

    虚拟机会保证,1,2两条语句一定会在3之前执行,4,5两条语句一定会在3之后执行,但不能保证1,2或者4,5之间不进行重排序。

    在对volatile修饰的变量进行写操作时,虚拟机会向处理器发送一条Lock前缀的指令,把变量的值写到主内存中,从而保证内存可见性。该指令就像一个内存屏障(内存栅栏),保证指令重排序时,不把其后面的指令排到内存屏障前面,也不把其前面的指令排到内存屏障后面。

    4. volatile和synchronized区别:

    volatile不支持原子性,synchronized是同步操作,一定是原子的;

    volatile并发时不会阻塞,而synchronized会发生阻塞。

    相关文章

      网友评论

          本文标题:volatile关键字

          本文链接:https://www.haomeiwen.com/subject/opeukftx.html