preface
To learn about multithreading in Java, you must be familiar with the mechanism of the volatile keyword. I have recently read a number of blogs about volatile and gained a preliminary understanding of it.
What’s the use?
Volatile provides two primary functions for the variables it modifies
- visibility
- Prevents instruction reordering
This blog focuses on volatile visibility, and a future post on instruction reordering.
What is visibility?
A picture is worth a thousand words
- Each Thread has its own working memory (can be understood as each cook has his own iron pot)
- All threads share one main memory (all chefs in the restaurant share the same refrigerator)
- Each Thread will go to the main memory to obtain data before operating data (cooks will go to the refrigerator to get food ingredients before cooking).
- Thread: cook
- Working memory: iron pot
- Store&load: Store cooked food and pick up ingredients
- Main memory: refrigerator
Readers can consider the following situation: a restaurant came to a customer ordered a dish of braised pork, at this time there are two chefs (assuming that the chefs do not communicate with each other), because of the non-communication, so the two chefs opened the refrigerator to take out the ingredients and began to cook. In the end, there were two portions of braised pork in brown sauce, and the customer only wanted one. Why did this happen?
Because there is no visibility between chefs.
To put this situation in JAVA, thread A fetched A variable from main memory into working memory. Thread B did not put the variable back into main memory in time after the operation, so thread B fetched the variable expired and fetched the variable before thread A operated.
How to have visibility?
I’ll start with eight atomic operations between working and main memory defined in the Java memory model
- Lock: A variable that acts on main memory and identifies a variable as a thread-exclusive state.
- Unlock: A variable that operates on main memory. It releases a locked variable so that it can be locked by another thread.
- Read: a variable acting on main memory that transfers the value of a variable from main memory to the thread’s working memory for subsequent load action.
- Load: a variable operating on *** working memory that places the value of the variable from main memory for the read operation into a copy of the variable in working memory.
- Use: a variable that operates on *** in-working memory. It passes the value of a variable in working memory to the execution engine. This operation is performed whenever the virtual machine reaches a bytecode instruction that needs to use the value of the variable.
- Assign: Applies to a variable in *** working memory. It assigns a value received from the execution engine to a variable in working memory. This operation is performed whenever the virtual machine accesses a bytecode instruction that assigns a value to the variable.
- Store: variable applied to *** working memory that transfers the value of a variable in working memory to main memory for subsequent write operations
- Write: a variable that operates on main memory and puts the value of the store operation from working memory into a variable in main memory.
Read the assignment of a common variable
- What’s the harm?
Let’s say I have one in main memory
int a=0;
Copy the code
Thread 1 and thread 2 execute once, ideally with a final value of 2.
a++;
Copy the code
Thread 1 after the assign operation on the real value of the variable a is from 0 to 1, but this process occurs in the working memory is not visible to other threads, if the thread 2 at this time to the operation of the variable a, read the value of the still is zero, since there is no visibility, thread 2 operation is merely repeating the thread 1 operation, again let a from 0 to 1. I didn’t get the desired a=2.
Read the assignment of a volatile variable
- Use cannot be read&load before
- Assign must be followed by Store&write
That is, read-load-use and assign-store-write become two indivisible atomic operations
Although there is still a vacuum between use and assign, and the variable may be read by another thread, the main memory variable and the working memory variable are equal at any point in time. This makes volatile variables unsuitable for operations that depend on the current value, such as increment. So where can volatile, which relies on visibility, be used? The Java Virtual Machine says:
The result of the operation does not depend on the current value of the variable (that is, the result does not depend on producing an intermediate result), or can ensure that only a single thread changes the value of the variable
Usually volatile is used to store a Boolean value for a state.
Partly referenced from
- Differences between volatile variables and common variables
- << In-depth understanding of Advanced Java Virtual Machine features and best practices >>