zoukankan      html  css  js  c++  java
  • java vloatile 关键字

    java中volatile关键字的使用涉及到java的内存模型,JMM。简单来说,每个线程都有一个本地内存(虚拟的),线程共享的变量存储在主内存中,主内存在堆中。内存访问方式如下图所示。

    volatile的作用总结如下:

    (1)可见性:对一个volatile变量的读取,总是能够看到(任意线程)对这个变量的最后写入,即能够读到最新的值。

    (2)原子性:针对vloatile基本类型(包括long double int等等)变量的读/写c操作具有原子性,但是复合操作不具有原子性。

    原子性操作的定义:原子性操作指的是一旦操作开始,就不能被线程调度中断,直到该原子操作执行完毕才会发生线程切换,所以不可分的原子操作就可以避免线程竞争,因此可以代替同步。但是除非是专家,一般我们都应该使用同步而不是原子操作。

    注意:对出long和double外的其他基本数据类型的读写操作都是原子性操作。但是JVM会将64位的long和double类型数据分开两个32为数据来进行读写操作,所以在读取或者写入中间会发生线程切换,从而导致数据破坏。因此需要在前面加上volatile关键字就可以保证long和double的存取也是原子性操作(atomic operation)。

      最后,什么时候使用volatile关键字呢?基本上,如果一个域可能被多个线程同时访问,并且这些线程中至少有一个是有对这个域做写操作的,那么就要把这个域设置为volatile的,  如果一个共享变量会被多个线程同时访问,那么使用volatile关键字可以保证数据的可视性,也就是在一个线程中更改了volatile类型域后,会立刻刷新主内存,其他线程读的时候读到的就是最新被修改的值。但是如果一个域已经完全由synchronized方法或者语句块保护,那么就不需要设置成volatile了,因为synchronized方法也会导致主内存的刷新。再次说明的是,第一选择应该是使用synchronized关键字,其他的方法都是有风险的。volatile的最主要作用是实现了多个线程之间数据的可视性。

      

    上述图中的操作过程描述

    (1)线程A写:线程A中更新了N的值,flag的值,对N的更新只是在本地内存中,但是对flag写入后,因为flag是volatile的,所以会导致本地内存A中被线程A更新过的两个共享变量被刷新到了主内存中,这样保证主内存中保存的是volatile型变量最新被写入的值。

    (2)线程B读:线程B读volatile类型的变量时,JMM会把本地内存置为无效,线程接下来将从主内存中读取共享变量。这样保证读取的是主内存中最新的值。

  • 相关阅读:
    Kth Largest Element in an Array -- LeetCode
    First Missing Positive -- LeetCode
    Path Sum II (Find Path in Tree) -- LeetCode
    Email List
    Divide Two Integers -- LeetCode
    Delete Node in a Linked List
    Compare Version Numbers -- LeetCode
    Broken Code
    Trapping Rain Water (Bar Height) -- LeetCode
    Count and Say (Array Length Encoding) -- LeetCode
  • 原文地址:https://www.cnblogs.com/wll-zju/p/4337839.html
Copyright © 2011-2022 走看看