- Small knowledge, big challenge! This article is participating in the creation activity of “Essential Tips for Programmers”.
- This article has participated in the “Digitalstar Project” and won a creative gift package to challenge the creative incentive money.
Preface:
AtomicInteger
The schwa ˈ t ɑ ː m ɪ k] the [ˈ nt ɪ d ʒ ɪ goes r]
There are three main features to consider in Java concurrency: atomicity, visibility, and orderliness. Synchronized keyword can guarantee visibility and order but not atomicity. And the purpose of this AtomicInteger is to ensure atomicity.
Why does this class exist?
Java concurrency atom class AtomicInteger
Because under concurrency, when I print i++
For i++ operations, there are actually three steps.
(1) Read the value of A from main memory
(2) Add 1 to a
(3) Refresh A to main memory
Some threads had incremented a by 1, but before they could reload to main memory, other threads re-read the old value. Because that’s what made the mistake.
Static AtomicInteger a = new AtomicInteger()
To print, use a.incrementandGet ()
This problem does not occur with AtomicInteger.
1. AtomicInteger method analysis:
There are a few ways to look at it:
Int getAndIncrement gets the old value and then increments it
Int incrementAndGet increments and then gets new values
What is the difference between the two methods?
1.1 getAndIncrement ()
Get current, add, compareAndSet(current, next) loop to return current if successful.
For JDK7, Java Concurrency Implementation Principles is also based on JDK7:
public final int getAndIncrement() { for (; ;) { int current = get(); int next = current + 1; if (compareAndSet(current, next))//CAS return current; } } public final boolean compareAndSet(int expect, int update) { return unsafe.compareAndSwapInt(this, valueOffset, expect, update); }Copy the code
For JDK8:
/** * Atomically increments by one the current value. * * @return the previous value */ public final int getAndIncrement() { return unsafe.getAndAddInt(this, valueOffset, 1); } public final int getAndAddInt(Object var1, long var2, int var4) { int var5; do { var5 = this.getIntVolatile(var1, var2); } while(! this.compareAndSwapInt(var1, var2, var5, var5 + var4)); return var5; } public final int getAndAddInt(Object obj, long valueOffset, int delta) {int expect; // get the main memory value expect = this.getIntVolatile(obj, valueOffset); } while(! this.compareAndSwapInt(obj, valueOffset, expect, expect + delta)); // return the old value expect; }Copy the code
Encapsulates the spin lock
Encapsulating the unsafe method directly to ensure atomicity, unsafe is JDK private and cannot be called
The main difference between JDK7 and 8 is that JAVA8 puts all loops in the Unsafe class.
1.2 incrementAndGet ()
public final int incrementAndGet() {
return unsafe.getAndAddInt(this, valueOffset, 1) + 1;
}
Copy the code
1.3 Difference between getAndIncrement and incrementAndGet
The first one returns the current value, and then +1;
Plus one and return the new value.
public static void main(String[] args) {
AtomicInteger first = new AtomicInteger(5);
int a = first.getAndIncrement();
System.out.println("a value:"+a);
AtomicInteger second = new AtomicInteger(5);
int b = second.incrementAndGet();
System.out.println("b value:"+b);
}
Copy the code
Output:
a value:5
b value:6
2. To summarize
When the interviewer asks, “Have you read the source code?
For AtomicInteger, you can mention the CAS and Unsafe classes.
The advantages of AtomicInteger
1. Optimistic locking, with strong performance, ensures atomicity by utilizing CPU’s own characteristics, that is, the CPU’s instruction set encapsulates compare and swap into one instruction to ensure atomicity.
2. Suitable for reading more and writing less mode
But the disadvantages are obvious
1. Spin consumes CPU performance. Therefore, sync is recommended for write operations
The JDK provides an AtomicStampedReference and an AtomicMarkableReference to resolve ABA problems. If the value of a spinner is changed by another thread, then the value of the spinner is changed again. Provides base data type and reference data type version number support
2.1 Additional: What are ABA problems?
The CAS operation will determine whether the value of V is still A. In some algorithms, CAS will succeed if the value of V changes first from A to B and then from B to A
2.2 CAS Implementation in Java
Cas in JAVA primarily uses the Unsafe method, where cas operations are primarily assembly instructions based on hardware platforms
Each time the CAS operation is performed, the thread retrieves the current value from memory according to valueOffset and compares it with expect’s value. If the value is consistent, it changes and returns true. If the value is inconsistent, another thread is changing the value of this object, and returns false
2.3 How to solve ABA?
The simplest solution to ABA is to add a modified version number to the value. Each time the value changes, the version number is changed and the CAS operation is compared to this version number.
2.4 Solutions to ABA problems in JAVA?
AtomicStampedReference maintains a pair containing an object reference and an integer “stamp” that can be automatically updated to solve ABA problems.
Public class AtomicStampedReference<V> {private static class Pair<T> {final T reference; // The maintenance object references final int stamp; Private Pair(T reference, int stamp) {this.reference = reference; this.stamp = stamp; } static <T> Pair<T> of(T reference, int stamp) { return new Pair<T>(reference, stamp); } } private volatile Pair<V> pair; . /** * expectedReference: The original value is updated * newReference: the new value to be updated * expectedStamp: the flag version you expect to update * newStamp: */ public Boolean compareAndSet(V expectedReference, V newReference, int expectedStamp, int newStamp) { Pair<V> current = pair; // Get the current pair return expectedReference == current. Reference && // The original value is equal to the value reference of the current pair, The value is unchanged The original mark version is equal to the current mark version of the pair, Mark did not change ((newReference = = current. The reference && newStamp = = current. The stamp) | | / / no change to update values and tag casPair (current, Pair.of(newReference, newStamp))); // Cas update pair}}Copy the code
How to use this class:
private static AtomicStampedReference<Integer> atomicStampedRef = new AtomicStampedReference<>(1, 0); Public static void main(String[] args){Thread main = new Thread(() -> {system.out.println (" Thread "+) Thread. CurrentThread () + ", the initial value a = "+ atomicStampedRef. GetReference ()); int stamp = atomicStampedRef.getStamp(); Thread.sleep(1000); thread.sleep (1000); // Wait 1 second for the interfering thread to execute} catch (InterruptedException e) {e.printStackTrace(); } Boolean isCASSuccess = atomicStampedRef.com pareAndSet (1, 2, stamp, stamp + 1); System.out.println(" operation Thread "+ thread.currentThread () +") CAS result " + isCASSuccess); }," main operation thread "); Thread other = new Thread(() -> { Thread.yield(); / / make sure that the thread - the main priority to perform atomicStampedRef.com pareAndSet (1, 2, atomicStampedRef getStamp (), atomicStampedRef. GetStamp () + 1); System. Out. Println (" Thread "+ Thread. CurrentThread () +", "increment" value = "+ atomicStampedRef. GetReference ()); AtomicStampedRef.com pareAndSet (2, 1, atomicStampedRef getStamp (), atomicStampedRef. GetStamp () + 1); System. The out. Println (" Thread "+ Thread. CurrentThread () +", "decrement" value = "+ atomicStampedRef. GetReference ()); }," thread interference "); main.start(); other.start(); }Copy the code
Operation results:
Rement (); rement (); rement (); rement (); rement (); rement (); rement ( Thread[main Thread,5,main],CAS result: false
Reference: ABA problems and solutions
Reference: blog.csdn.net/fenglllle/a…