Does Java volatile prevent caching or enforce write-through caching?

后端 未结 2 1008
暖寄归人
暖寄归人 2021-02-05 15:17

I\'m trying to understand Java\'s volatile keyword with respect to writing to a volatile atomic variable in a multithreaded program with CPU caches

2条回答
  •  [愿得一人]
    2021-02-05 15:35

    Within Java, it's most accurate to say that all threads will see the most recent write to a volatile field, along with any writes which preceded that volatile read/write.

    Within the Java abstraction, this is functionally equivalent to the volatile fields being read/written from shared memory (but this isn't strictly accurate at a lower level).


    At a much lower level than is relevant to Java; in modern hardware, any and all reads/writes to any and all memory addresses always occur in L1 and registers first. That being said, Java is designed to hide this kind of low level behavior from the programmer, so this is only conceptually relevant to the discussion.

    When we use the volatile keyword on a field in Java, this simply tells the compiler to insert something known as a memory barrier on the reads/writes to this field. A memory barrier effectively ensures two things;

    1. Any threads reading this address will use the most up-to-date value (the barrier makes them wait until the most recent write makes it back to shared memory, and no reading threads can continue until this updated value makes it to their L1 cache).

    2. No reads/writes to ANY fields can cross over the barrier (aka, they are always written back before the other thread can continue, and the compiler/OOO cannot move them to a point after the barrier).

    To give a simple Java example;

    //on one thread
    counter += 1; //normal int field
    flag = true; //flag is volatile
    
    //on another thread
    if (flag) foo(counter); //will see the incremented value
    

    Essentially, when setting flag to true, we create a memory barrier. When Thread #2 tries to read this field, it runs into our barrier and waits for the new value to arrive. At the same time, the CPU ensures that counter += 1 is written back before that new value arrives. As a result, if flag == true then counter will have been incremented.


    So to sum up;

    1. All threads see the most up-to-date values of volatile fields (which can be loosely described as "reads/writes go through shared memory").

    2. Reads/writes to volatile fields establish happens-before relationships with previous reads/writes to any fields on one thread.

提交回复
热议问题