• volatile简介


    volatile简介

    java语言提供了一种稍弱的内存同步机制,即volatile变量。用来确保将变量的更新操作通知到其它线程,保证了新值能立即同步到主内存,以及每次使用前立即从内存刷新。当变量声明为volatile类型后,编译器与运行时都会注意到这个变量是共享的。

    volatile变量对所有线程可见,对volatile变量所有的写操作都能立即反应到其它的线程中,但是volatile并不能保证操作的原子性,因此volatile变量的运算并不是线程安全的。

    并发编程中的三个概念

    在并发编程中,我们通常会遇到以下三个问题:原子问题,可见性问题,有序性问题。

    原子性:

    即一个操作或多个操作要么全部执行并且执行过程中不能被任何操作打断,要么都不执行。只有简单的读取、赋值(而且必须是将数字赋值给某个变量,变量 之间的赋值不是原子操作)才是原子操作。java内存模型只保证了基本读取和赋值是原子操作,如果要实现更大范围操作的原子性,可以通过 synchronized和lock来实现。由于synchronized和lock能保证任一时刻只有一个线程执行该程序块,那么自然就不存在原子性问 题了,从而保证了原子性。

    可见性:

    可见性是指多个线程访问同一个变量时,一个线程修改了这个变量的值,其它线程能够立即看到修改的值。对于可见性,java提供了volatile关 键字来保证可见性。当一个共享变量被volatile修饰的时候,它会保证修改的值立即被更新到内存,当有其它线程需要读取时,它会去内存中去读新值。而 普通共享变量不能保证可见性。因为普通共享变量被修改后,什么时候被写入内存是不确定的。当其他线程去读取新值时,此时内存中可能还是原本的值,因此无法 保证可见性。

    有序性:

    即程序执行的顺序按照代码的先后顺序执行。在java内存模型中,允许编译器和处理器对执行进行重排,但是重排过程不会影响单线程程序的执行,却会 影响到多线程并发执行的正确性。在java中,可以通过volatile关键字来保证一定的有序性。另外可以通过synchronized和lock来保 证有序性。很显然,synchronized和lock保证了每个时刻只有一个线程执行同步代码,相当于是让线程执行同步代码,自然保证了有序性。

    指令重排:

    一般来说,处理器为了提高程序运行效率,可能会对输入代码进行优化,它不保证程序中各个语句的执行顺序同代码中的顺序一致,但是它会保证程序最终执 行结果和代码顺序执行的结果是一致的。如果两个语句之间没有数据依赖,那么可能会被重排。指令重排不会影响单个线程的执行,但是会影响到线程并发执行的正 确性。

    在并发编程中,如果想要一个程序正确的执行,必须保证原子性、可见性以及有序性,只要有一个没有被保证,就有可能会导致程序运行不正确。

    volatile关键字的两层含义

    一旦一个共享变量(类的成员变量、类的静态成员变量)被volatile修饰之后,那么就具备了两层含义:

    1)保证了不同线程对这个变量进行操作时的可见性,即一个线程修改了某个变量的值,这个新值对其它线程来说是立即可见的。

    2)禁止指令重排。

    volatile关键字能够保证可见性,但是不能保证原子性。

    volatile能禁止指令重排,所以volatile能在一定程度上保证有序性。

    volatile关键字机制指令重排有两层意思:

    1.当程序执行到volatile变量的读操作或者写操作时,在其前面的操作的更改肯定全部已经执行结束,并且结果已经对后面的修改可见,其后的操作一定还没有进行。

    2.在进行指令优化时,不会对volatile变量进行重排。

    从volatile关键字生成的汇编代码发现,加入volatile关键字时,会多出一个lock前缀指令。

    lock前缀指令实际上相当于一个内存屏障。内存屏障会提供三个功能:

    1)它确保指令重排时不会把其后面的指令排到内存屏障之前的位置,也不会把前面的指令排到内存屏障之后,即在执行到内存屏障这句指令时,在它前面的操作已经全部完成。

    2)它会强制对缓存的修改操作立即写入内存。

    3)如果是写操作,它会导致其他cpu中对应的缓存行无效。

    使用volatile关键字的场景

    synchronized关键字是防止多个线程同时执行一段代码,那么就会影响到程序的执行效率,而volatile关键字在某些情况下性能要优于 synchronized,但是要注意volatile关键字无法保证操作的原子性。通常来说,使用volatile必须具备以下两个条件。

    1)对变量的写操作不依赖于当前值。

    2)该变量没有包含在具有其它变量的不变式中。

    意思就是保证操作是原子操作,才能保证使用volatile关键字的程序在并发时能够正确执行。

  • 相关阅读:
    docker node中uid与gid的授权问题
    windows下docker无法进行端口映射的问题
    IOS/Safari下document对象的scrollHeight值比Chrome更大
    Vue/Egg大型项目开发(二)数据库设计
    .babelrc和babel.config.js的相同配置不能合并
    es6 class中责任链模式与AOP结合
    JS设计模式(10)职责链模式(重要)
    Vue/Egg大型项目开发(一)搭建项目
    你不知道的JS(3)来聊聊this
    CentOS7为php7.2安装php-redis扩展(redis环境搭建二)
  • 原文地址:https://www.cnblogs.com/benniaoxuefei/p/5489312.html
Copyright © 2020-2023  润新知