“This is my 27th day of participating in the First Challenge 2022. For more details: First Challenge 2022.”
The principle of exclusive acquisition and release of synchronous queue and synchronous state in AQS is introduced in detail.
AQS Related articles:
AQS (AbstractQueuedSynchronizer) source depth resolution (1) – design and general structure of AQS
AQS (AbstractQueuedSynchronizer) source depth resolution (2) – Lock interfaces and implementation of custom Lock
AQS (AbstractQueuedSynchronizer) source depth resolution (3) – synchronous queue and exclusive access to the principle of lock, lock is released [ten thousand words]
AQS (AbstractQueuedSynchronizer) source depth resolution (4), the principle of Shared locks, lock is released [ten thousand words]
AQS (AbstractQueuedSynchronizer) source depth resolution (5) – condition queue waiting, the realization of the notification and summary of AQS [ten thousand words]
The synchronization queue in AQS is closely related to the acquisition, release and blocking of the synchronization state, and these two knowledge points must be studied together.
1 Synchronization queue structure
public abstract class AbstractQueuedSynchronizer extends
AbstractOwnableSynchronizer implements java.io.Serializable {
/** * The thread that currently acquires the lock. This variable is defined in the parent class, which AQS inherits directly. When acquiring an exclusive lock, if it is a reentrant lock, you need to know which thread acquired the lock. Nothing is null */
private transient Thread exclusiveOwnerThread;
/** * The thread attribute of the Node to which the head points is always null. * /
private transient volatile Node head;
/** * the end of the queue, all subsequent nodes are added to the end of the queue */
private transient volatile Node tail;
/**
* εζ₯ηΆζ
*/
private volatile int state;
/** * Node internal class, synchronization queue Node type */
static final class Node {
/*AQS supports both shared mode and exclusive mode. The following represents the constructed node type flag */
/** * A node constructed in shared mode, which is used to mark the thread as the */ of the AQS queue that has been blocked and suspended while fetching shared resources
static final Node SHARED = new Node();
/** ** a node constructed in exclusive mode to mark the thread as the */ that was blocked and suspended from the AQS queue while obtaining exclusive resources
static final Node EXCLUSIVE = null;
/* The wait state of the thread node, which is used to indicate the state of the wait lock on the thread */
/** * indicates that the current node (thread) needs to cancel the wait * because the thread waiting in the synchronization queue has timeout, interrupt, exception, i.e. has abandoned the lock, needs to cancel the wait from the synchronization queue */ if the node enters this state, it will not change to other states */
static final int CANCELLED = 1;
/** * Indicates that subsequent nodes (threads) of the current node (thread) need to unwait (wake up) * if a node state is set to SIGNAL, the subsequent threads of the current node are suspended or about to suspend. An attempt will be made to wake up the thread of the successor to run *, a state that is usually set by the successor to its predecessor. When a thread of a node is about to be suspended, an attempt is made to set the state of the precursor node to SIGNAL */
static final int SIGNAL = -1;
/** * The thread is waiting on Condition. The value waitStatus indicates that the thread is waiting on Condition. When the Condition is called signal() by another thread, the node is moved from the wait queue to the synchronous queue to obtain the synchronous state */
static final int CONDITION = -2;
/** * Other nodes need to be notified when a shared resource is released. The waitStatus value indicates that the next shared synchronization state acquisition should be propagated unconditionally */
static final int PROPAGATE = -3;
/** * Records the current thread wait state value, including the state in 4 above, and 0, indicating the initialization state */
volatile int waitStatus;
/** ** the information of the parent node will be set when the parent node joins the synchronization queue */
volatile Node prev;
/** ** the next node */
volatile Node next;
/** * The thread that currently gets the synchronization status */
volatile Thread thread;
/** * This field is a SHARED constant * that is always null in exclusive lock mode. * /
Node nextWaiter;
/** * Returns true if the Node nextWaiter field is a SHARED constant in SHARED mode */
final boolean isShared(a) {
return nextWaiter == SHARED;
}
/** * is used to create the initial header or SHARED tag */
Node() {
}
/** * used to add to wait queue **@param thread
* @param mode
*/
Node(Thread thread, Node mode) {
this.nextWaiter = mode;
this.thread = thread;
}
/ /...}}}Copy the code
From the above source code, the basic structure of the synchronization queue is shown as follows:
In the source code of AQS internal Node, we can see that the synchronization queue is a variant of “CLH” (Craig, Landin, andHagersten) lock queue. Its head reference points to the head Node as the sentinel Node, which does not store any information related to the waiting thread, or can be regarded as the Node that has obtained the lock. The second node starts out as the actual node for the thread to build, and subsequent nodes are added to the end of the list.
The way to add new nodes to the end of a linked list is the compareAndSetTail(Node Expect,Node Update) method, which is a CAS method that keeps threads safe.
The node where the thread finally obtains the lock will be set as the head node (setHead method). This setting step is accomplished by the thread that has successfully acquired the lock. Since only one thread can successfully acquire the lock, CAS is not required to ensure this setting method.
Synchronous queues follow a first-in, first-out (FIFO), where the next node of the head node is the node that will acquire the lock, and the thread will wake up the successor when it releases the lock, which will then attempt to acquire the lock.
2 Lock acquisition and release
The state of the Lock is represented by the state variable. Generally, 0 indicates that the Lock is not occupied, and greater than 0 indicates that the Lock is occupied.
The lock acquisition and release provided by AQS can be divided into exclusive and shared locks:
- Exclusive: As the name implies, only one thread can acquire the lock at a time, and other threads can wait in the synchronization queue. Only when the thread that acquires the lock releases the lock, the subsequent thread can acquire the lock.
- Shared: Multiple threads can acquire locks at the same time.
For AQS, the key to thread synchronization is the operation on the synchronization state:
- To obtain and release locks in exclusive mode, use the following methods:
Void acquire(int arg), void acquirelnterruptibly(int arg), Boolean Release (int arg)
. - Locks can be acquired and released in shared mode by:
Void acquireShared (int arg), void acquireSharedInterruptibly (int arg), Boolean reaseShared (int arg)
.
The general process for obtaining locks is as follows:
The thread first tries to acquire the lock, and if it fails, it adds the current thread and wait state to the synchronization queue as a Node. If the lock fails, it will attempt to block itself (if the current node’s precursor is in SIGNAL state). After blocking, no subsequent code will be executed until it is woken up. When the thread holding the lock releases the lock, it wakes up a successor thread in the queue, or if the blocking thread is interrupted or time runs out, the blocking thread is also woken up.
If there are exclusive and shared, then there are these differences under the general steps above:
- The exclusive lock is tied to a specific thread. If one thread has acquired the lock, the ‘exclusiveOwnerThread’ field records the thread. When another thread attempts to acquire the lock by using the ‘State’ operation, it will find that it does not own the lock and will be placed in the AQS synchronization queue. For example, in the implementation of the exclusive ReentrantLock, when a thread acquires the ReentrantLock, it will first use CAS operation to change the state value from 0 to 1 in AQS, and then set the holder of the current lock as the current thread. When the thread acquires the lock again and finds that it is the holder of the lock, the status value will be changed from 1 to 2, that is, the number of reentrant times will be set. When another thread acquires the lock and finds that it is not the holder of the lock, it will be put into the AQS synchronization queue and suspended.
- When multiple threads apply for the lock, they compete for the lock through CAS. When one thread acquires the lock, another thread acquires the lock again. If the current lock still meets its needs, the current thread only needs to acquire the lock through CAS. For example, Semaphore Semaphore, when a thread acquires Semaphore through acquire () method, it will first check whether the current number of Semaphore meets the need, if not, it will put the current thread into the synchronization queue, if so, it will acquire the Semaphore through spin CAS, and the corresponding number of Semaphore decreases the corresponding value.
In fact, the specific steps are more complex, which will be mentioned in the source code below!
3 Acquire Exclusive acquire lock
The acquire lock can be obtained exclusively by calling the ACQUIRE template method of AQS. This method will not respond to interrupt, that is, the thread will not be removed from the synchronization queue after it fails to obtain the synchronization status. Components based on an exclusive implementation include ReentrantLock.
The general steps of the method are as follows:
- The tryAcquire method is first called to attempt to acquire the lock. If the lock is acquired successfully, true is returned, and the method ends. Otherwise, return false and proceed to the next step.
- Construct a synchronization Node using the addWaiter method in node. EXCLUSIVE mode and add the thread to the end of the synchronization queue.
- Then continue spinning to acquire the lock by acquireQueued(Node Node,int arg).
- If the lock is not acquired in a spin, then determine if it can be suspended and try to suspend the thread in the node (call locksupport.park (this) to suspend yourself, note that the thread state is WAITING). The suspended thread is awakened mainly by the interruption of the precursor node or thread. Note that it will continue to spin to try to acquire the lock after awakening.
- Finally, only the thread that acquired the lock can return from the acquireQueued method, and the return value is used to determine whether selfInterrupt is called to set the interrupt flag, but the thread is running and will not raise an exception even if the interrupt flag is set (that is, the acquire (Lock) method will not respond to the interrupt).
- The thread acquires the lock, the acquire method terminates, the lock method returns, and the synchronization code continues!
/** ** an exclusive attempt to obtain the lock, but fails to obtain the lock until it enters the synchronization queue
public final void acquire(int arg) {
// The interior is made up of four method calls
if(! tryAcquire(arg) && acquireQueued(addWaiter(Node.EXCLUSIVE), arg)) selfInterrupt(); }Copy the code
3.1 tryAcquire attempted to acquire an exclusive lock
The familiar tryAcquire method, which we mentioned at the beginning of “Design of AQS”, is a subclass of AQS, which we implemented ourselves, and is used for the first attempt to acquire an exclusive lock. Generally, this means changing the state, or checking for reentrant locks, setting the thread currently acquiring the lock, etc. Different locks have their own logical judgments, which will not be covered here, but will be covered later when we talk about the implementation of specific locks (such as ReentrantLock). In general, this method returns true on success and false on failure.
In AQS, tryAcquire’s implementation throws an exception, so subclass override is required:
protected boolean tryAcquire(int arg) {
throw new UnsupportedOperationException();
}
Copy the code
3.2 addWaiter Joins the Synchronization queue
The addWaiter method is provided by AQS and does not require us to rewrite, or lock, the generic method!
The addWaiter method is used to add node. EXCLUSIVE, a synchronization Node constructed in EXCLUSIVE mode, to the end of the synchronization queue. The general steps are as follows:
- Build a new node according to the given pattern.
- If the synchronization queue is not null, a new node is added to the end of the queue (only once). If it succeeds, the new node is returned and the method ends.
- If the queue is null or fails to be added, the enQ method is called to loop until it succeeds, returns a new node, and the method ends.
/** * The addWaiter(Node Node) method adds the thread that failed to acquire the lock as a Node to the end of the synchronization queue **@paramMode model. EXCLUSIVE mode passes a Node.EXCLUSIVE (null). SHARED mode passes in a Node.shared, which is a static Node object (SHARED, same) *@returnReturns the constructed node */
private Node addWaiter(Node mode) {
/*1 first constructs the node */
Node node = new Node(Thread.currentThread(), mode);
/*2 Try to place the node directly at the end of the queue */
// Get the tail node of the synchronizer directly and use pred to save it
Node pred = tail;
/* If pred is not null, which means queue is not null * then CAS is used to set the current node to the end node * */
if(pred ! =null) {
node.prev = pred;
// The CAS method of compareAndSetTail is used to ensure that the node can be added safely by the thread, although it may not succeed.
if (compareAndSetTail(pred, node)) {
// place the newly constructed node as the successor to the end of the original team
pred.next = node;
// return the new node
returnnode; }}* (1) the queue may be null * (2) the enq method is called to ensure that the constructed node is successfully added to the synchronous queue * */
enq(node);
return node;
}
/** * Node constructor ** used in the addWaiter method@paramThread Current thread *@paramMode pattern * /
Node(Thread thread, AbstractQueuedSynchronizer.Node mode) {
// The successor node in the wait queue is equal to the mode of that node
In SHARED mode, the value is the constant of Node.SHARED mode, the value is null
this.nextWaiter = mode;
// The current thread
this.thread = thread;
}
Copy the code
3.2.1 ENQ ensures that nodes join the queue
The enq method is used when the synchronization queue is null or a CAS addition fails. Enq ensures that the node will eventually be added successfully. The general steps are as follows:
- Start an infinite loop, in which the following operations are performed;
- If the queue is empty, initialize the queue, add a sentinel node, end the loop, continue the next loop;
- If the queue is not empty, then the previous method is the same, then try to add the new node to the end of the queue, if successful, return the new node’s precursor, the end of the loop; If not, end the loop and continue the next one.
The enq method returns the precursor of the new node, which is not used in the addWaiter method.
Also, the compareAndSetHead method for adding headers and the compareAndSetTail method for adding endpoints are both CAS methods, and both call native methods in the Unsafe class, because threads hanging, recovery, CAS operations, and so on are ultimately implemented in the operating system. Unsafe class provides a Java directly interface for interacting with the underlying operating system, the inside of a class of many operation is similar to the C pointer manipulation, by finding the offset of a property, directly to the property assignment, for docking with the Java native methods are Hospot source code in the method, and these methods are written using C + +, Pointers must be used!
Unsafe is also the cornerstone of AQS concurrency control mechanism. Looking Unsafe, you can first learn about AQS.The Unsafe class in Java explains its principles and use cases.
/** * loop until the end node is successfully added */
private Node enq(final Node node) {
/* Operate in an infinite loop until successfully added */
for(; ;) {// get the end node t
Node t = tail;
/* If the queue is null, initialize the synchronization queue */
if (t == null) {
/* Call compareAndSetHead to initialize the synchronization queue * note: this is a new blank node, which is the legendary sentinel node * after the CAS is successful, head will point to the sentinel node and return true * */
if (compareAndSetHead(new Node()))
// The end points to the head (sentry)
tail = head;
/* instead of ending, the loop continues, at which point the queue is no longer empty, so the following logic */ occurs
}
/* If the queue is not null, call compareAndSetTail and add a new node to the end of the queue
else {
/*1 first changes the direction of the new node's precursor. This step is not safe, but it doesn't matter, because if there is a conflict in this step, then the following CAS operation must succeed and the other threads will retry */
node.prev = t;
/* * 2 Call compareAndSetTail (CAS) to try to add a node to the end of the synchronous queue
if (compareAndSetTail(t, node)) {
//3 Change the direction of the parent node's successors
t.next = node;
// return a new node to end the loop
returnt; }}}}/** * CAS adds a header. * * is used only in the ENQ method@paramUpdate header *@returnTrue on success; * / or false on failure
private final boolean compareAndSetHead(Node update) {
return unsafe.compareAndSwapObject(this, headOffset, null, update);
}
/** * CAS adds a tail. * * is used only in the ENQ method@paramExpect expects the original endpoint *@paramUpdate the new endpoint *@returnTrue on success; * / or false on failure
private final boolean compareAndSetTail(Node expect, Node update) {
return unsafe.compareAndSwapObject(this, tailOffset, expect, update);
}
Copy the code
In the addWaiter and ENQ methods, there are three steps to becoming a tail:
- Set the front drive PREV
- Set up the tail
- Set the successor next
Since tail is the CAS operation, we can only guarantee that the prev of node is correct. However, we can not set the next thread successfully and switch to another thread immediately. In this case, next may be null, but its successor may not be null.
So synchronous queues only guarantee that the precursor prev is reliable, but next is not, so the rest of the source code is basically traversed backwards through the precursor PREV.
3.3 acquireQueued Node spin lock
If you can reach this method, the tryAcquire() and addWaiter() methods are passed, indicating that the thread has failed to acquire the lock and is placed at the end of the synchronization queue.
AcquireQueued method said node action after entering the synchronous queue, in fact is the process of entering a spin, spin in the process, when the condition is met, get into the lock, you can exit and return from the spin, otherwise it may block the nodes of the thread, subsequent even if the block is awakened, will spin attempts to acquire locks, Until success or an exception is thrown.
Finally, if the method exits because of a lock, it returns a flag bit that indicates no break or exits because of an exception, and throws an exception! The general steps are as follows:
- Also starts an infinite loop, in which the following operations are performed;
- If the head node is the precursor of the current node, it tries to acquire the lock. If the lock is obtained successfully, the current node is set as head node. The thread of the current node exits the queue, indicating that the current thread has obtained the lock.
- If the current node’s predecessor is not the head node or attempts to acquire the lock fail, then determine whether the current thread should be suspended. If true, then call parkAndCheckInterrupt to suspend the current node’s thread (locksupport. park suspends the thread). The thread is WAITING, and no further steps or code will be executed.
- If the current thread should not be suspended, that is, false is returned, the loop ends and the next loop continues.
- If the thread is woken up by another thread, determine if it was woken up because of an interrupt and modify the flag bit, while continuing the loop until the lock is acquired in Step 2. (This is also how acquire methods do not respond to interrupts – the Park method does not throw an exception when it is interrupted, just returns from the pending state and then needs to continue trying to acquire the lock)
- Finally, if the thread gets the lock and jumps out of the loop, or if an exception occurs, the finally statement block will be executed. In finally, it can determine whether the thread jumps out of the loop because of an exception. If so, cancelAcquire method will be executed to cancel the node’s request for obtaining the lock. If not, that is, out of the loop because the lock was acquired, then nothing will be done in finally!
/ * * *@paramNode New node *@paramArg parameter *@returnReturns true */ if interrupted while waiting
final boolean acquireQueued(final Node node, int arg) {
//failed indicates whether the lock failed to be obtained
boolean failed = true;
try {
// Interrupted indicates whether the alarm is interrupted
boolean interrupted = false;
/* infinite loop */
for(; ;) {// Get the precursor of the new node
final Node p = node.predecessor();
/* Only try to acquire the lock if the precursor is a header * */
if (p == head && tryAcquire(arg)) {
// After acquiring the lock, set itself as the head node (sentry node) and the thread exits the queue
setHead(node);
// The link of the precursor node (the original sentinel node) is null and is reclaimed by the JVM
p.next = null;
If the value is set to false, the lock is successfully obtained
failed = false;
// Returns interrupted, that is, whether the thread was interrupted
return interrupted;
}
/* The precursor is not a header or failed to get synchronization status */
/ * shouldParkAfterFailedAcquire detects whether the thread should be suspended, if the return true * the call parkAndCheckInterrupt used to thread hangs * or * * / restart cycle
if (shouldParkAfterFailedAcquire(p, node) &&
parkAndCheckInterrupt())
/* The current thread (thread) woke up because it was interrupted, so it changes its interrupt flag status to true * and starts the loop again until it has acquired the lock
interrupted = true; }}/* The finally statement block that is executed when a thread acquires a lock or an exception occurs finally {
/* If failed is true, it indicates that lock acquisition failed, that is, corresponding to the occurrence of exceptions. Here, exceptions may be thrown only in the tryAcquire method and predecessor method, and locks are not obtained at this time. Failed =true Then execute cancelAcquire method, which is used to cancel the thread's lock acquisition request, cancel the thread status of the node CANCELLED, and attempt to remove the node (if it is a tail node). This method is also called. If failed is false, the lock is obtained, and the method ends and continues. * /
if (failed)
// Cancel the lock request, remove the current node from the queue,cancelAcquire(node); }}/** * A method in a Node class returns the previous Node or raises a NullPointerException if null. Used when the current set cannot be null. The null check can be cancelled, indicating that this exception has no code-level significance, but can help the VM? So what does this exception really do? * *@returnThe precursor of this node */
final Node predecessor(a) throws NullPointerException {
// Get the precursor
Node p = prev;
// If null, an exception is thrown
if (p == null)
throw new NullPointerException();
else
// Return to the precursor
return p;
}
/** * head points to a new node. This method is called after tryAcquire has acquired the lock and does not raise thread-safety issues@paramNode New node */
private void setHead(Node node) {
head = node;
// The thread and prev attributes of the new node are null
// The new node becomes a sentinel node, and the internal thread exits the queue
// Set the thread reference to null, but in the tryAcquire method track the thread that acquired the lock, so don't worry about finding which thread acquired the lock
// It can also be seen that the sentinel node can also be called "the lock obtained node".
node.thread = null;
node.prev = null;
}
Copy the code
3.3.1 shouldParkAfterFailedAcquire node should be hung
ShouldParkAfterFailedAcquire method calls, after no access to the lock is used to determine whether the current node needs to be hung. The steps are as follows:
- If the precursor node is already SIGNAL(-1), the current node can be suspended.
- Otherwise, if the status of the precursor Node is greater than 0, Node.CANCELLED, it means that the precursor Node has given up the lock wait, then the precursor should be searched forward until a Node whose status is less than or equal to 0 is found, and the current Node is behind the Node. Return false, and the method ends.
- Otherwise, the state of the precursor is neither SIGNAL(-1) nor CANCELLED(1). CAS sets the status of the precursor to SIGNAL(-1) and returns false. End of the method!
The current node can be suspended only if its precursor is in SIGNAL state, otherwise it spins!
The SIGNAL state of a node is set by its successor. The SIGNAL state of a node is set by its successor. The SIGNAL state of a node is set by its successor.
/** * Checks whether the current node (thread) should be suspended **@paramPred The precursor * of this node@paramNode The node *@returnIf the driver is already SIGNAL, the current node can be suspended. Otherwise, it might look for a new precursor or try to set the precursor to SIGNAL, returning false */
private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) {
// Get the previously fetched waitStatus_ wait state
// Recall that when a node is created, no value is assigned to waitStatus, so each node starts with a value of 0 waitStatus
int ws = pred.waitStatus;
/* If the precursor node is already SIGNAL, the current node can be suspended */
if (ws == Node.SIGNAL)
return true;
/* If the status of the precursor Node is greater than 0, Node.CANCELLED indicates that the precursor Node has abandoned the lock wait */
if (ws > 0) {
/ * by the precursor look ahead, until we find a node of state is less than or equal to 0 (no cancellation of nodes), nodes become the current after the flooding, this step is very important, may clear the node of a has been cancelled, and if the precursor to release the lock, also will wake its successor, keep the queue activity * /
do {
node.prev = pred = pred.prev;
} while (pred.waitStatus > 0);
pred.next = node;
}
/* Otherwise, the state of the precursor is neither SIGNAL(-1) nor CANCELLED(1)*/
else {
/* The state CAS of the precursor node is set to SIGNAL(-1), which may fail, but it doesn't matter because the loop will continue */ after failure
compareAndSetWaitStatus(pred, ws, Node.SIGNAL);
}
// Return false to indicate that the current node cannot be suspended
return false;
}
Copy the code
3.3.2 parkAndCheckInterrupt Suspends the thread & determine the interrupt status
ShouldParkAfterFailedAcquire method returns true, after will call parkAndCheckInterrupt method hung thread interrupt status and subsequent judgment, in two steps:
- Use locksupport.park (this) to suspend the thread without further steps or code. Until the thread is interrupted or woken up (unpark)!
- If the Thread has been interrupted or woken up, returns the value of the thread.interrupted () method, which checks the interrupted status of the previous Thread and clears the interrupted status, that is, if the Thread was woken up because it was interrupted, the interrupted status is true, reset the interrupted status to false, and returns true. If the thread is not woken up because of an interrupt, the interrupt status is false and false is returned.
/** * Suspends the thread and returns the interrupted status when the thread returns@returnReturn true if due to thread interruption, false */ otherwise
private final boolean parkAndCheckInterrupt(a) {
/*1) Use locksupport. park(this) to suspend the thread without further steps or code. Until the thread is interrupted or woken up (unpark) */
LockSupport.park(this);
/*2) If the Thread has been interrupted or woken up, return the value of thread.interrupted () method, which checks the interrupted status of the previous Thread and clears the interrupted status, that is, if the Thread was woken up because it was interrupted, the interrupted status is true and the interrupted status is reset to false. And returns true. Notice that the park method does not throw an exception when it is interrupted! If the thread is not woken up because of an interrupt, the interrupt status is false and false*/ is returned
return Thread.interrupted();
}
Copy the code
3.3.3 Finally code block
In the acquireQueued method, you have a finally block that will execute no matter what happens in the try. In the acquire exclusive, uninterruptible acquire lock method, there are only two situations in which finally can be executed:
- When the current node (thread) finally acquires the lock, finally is entered and failed = false is set after the lock is acquired.
- When an exception occurs ina try, jump directly to finally. The exception that occurs here can only happen in the tryAcquire or predecessor methods and then directly into the finally code block, at this point the lock has not been acquired, failed=true!
- The tryAcquire method is implemented by ourselves and we decide what exceptions to throw. Even if an exception is thrown, it is not normally thrown in the acquireQueued and may be thrown when tryAcquire is called in the first place.
- The predecessor method checks that nullPointerExceptions are thrown if the predecessor is null. But the comment also says that this check has no code-level meaning, perhaps the exception will never be thrown?
The logic in the finally code block is:
- If failed = true, finally is performed without obtaining the lock, and an exception occurs. CancelAcquire cancels the lock request of the current node thread, acquireQueued ends, and throws an exception.
- If failed = false, the lock has been acquired, then nothing will actually be executed in finally. AcquireQueued method ends, and returns interrupted — if interrupted.
To sum up, in acquire exclusive uninterruptible lock acquisition methods, most of the situations in finally are nothing to do, or almost no exception is thrown, so cancelAcquire method is basically ignored.
However, it is quite common for cancelAcquire to be executed in interruptible or time-out lock acquisition methods. So put the cancelAcquire method source code analysis in the interruptible lock acquisition method source code analysis section!
3.4 selfInterrupt Security Interrupt
SelfInterrupt is the last method that can be called in acquire and, as its name suggests, is used for secure interrupts. TryAcquire and acquireQueued return values to determine whether the interrupt flag bit needs to be set.
The thread is interrupted only if tryAcquire attempts fail and acquireQueued is true. But the interrupt flag bit in parkAndCheckInterrupt is determined and then reset (the interrupted method resets the interrupt flag bit).
Although it seems useless, in a responsible manner, I will record the interrupt flag bit. Reset the thread’s interrupt flag bit to true.
/** * interrupts the current thread. The current thread is running, so only the interrupt flag is set, and no exception is raised
static void selfInterrupt(a) {
Thread.currentThread().interrupt();
}
Copy the code
4 release Release the exclusive lock
After the current thread has acquired the lock and executed the logic, it needs to release the lock so that subsequent nodes can continue to acquire the lock. AQS release(int arg) {AQS release(int arg);
- Try using tryRelease(arG), which we discussed at the beginning, as a self-implemented method, usually setting state to 0 or reducing or eliminating the thread currently holding the lock, etc., and return true if the lock is released on success, false otherwise;
- Return true if the tryRelease succeeded, determine that if the head is not null and the state of the head is not zero, then try calling the unparkprecursor method to wake up a non-cancelled successor to the head so that it can obtain the lock. Return true, the method ends;
- If tryRelease fails, return false and the method ends.
/** * exclusive release synchronization state **@paramArg parameter *@returnReturn true on success, false */ otherwise
public final boolean release(int arg) {
/*tryRelease (tryRelease) ΒΆ tryRelease (tryRelease) ΒΆ tryRelease (tryRelease) ΒΆ tryRelease (tryRelease
if (tryRelease(arg)) {
// Get the header
Node h = head;
// If the header is not null and the state is not zero
if(h ! =null&& h.waitStatus ! =0)
/* Then wake up a successor of the header in a wait-locked state * this method was described in acquire * */
unparkSuccessor(h);
return true;
}
return false;
}
Copy the code
4.1 Unparkprecursor awakens the successor
The unparkprecursor is used to wake up some non-cancelled successor of the parameter node, and is called in many ways, in the following steps:
- If the state of the current node is less than 0, the CAS is set to 0, indicating that subsequent nodes can continue to attempt to acquire the lock.
- If the successor S of the current node is null or CANCELLED, s will be pointed to NULL first. Then search backwards from tail to node and assign s to the nearest non-cancelled node. You need to traverse backwards because the synchronous queue only guarantees the correctness of the node’s precursor relationship.
- If s is not null, the state must not be CANCELLED. Instead, wake up the thread of S and call locksupport. unpark.
/** * wakes up the successor of the specified node **@paramNode specifies the node */
private void unparkSuccessor(Node node) {
int ws = node.waitStatus;
/* * 1) If the state of the current node is less than 0, then CAS is set to 0, indicating that subsequent node threads can try to acquire the lock first rather than suspend it directly. * * /
if (ws < 0)
compareAndSetWaitStatus(node, ws, 0);
// Get the immediate descendant of node
Node s = node.next;
/* * 2) If S is null or CANCELLED, search backwards from tail to node and assign s to the nearest non-cancelled node. * * /
if (s == null || s.waitStatus > 0) {
s = null;
for(Node t = tail; t ! =null&& t ! = node; t = t.prev)if (t.waitStatus <= 0)
s = t;
}
/* * 3) If s is not null, the state must not be CANCELLED. Instead, wake up the thread of S and call locksupport. unpark. * * /
if(s ! =null)
LockSupport.unpark(s.thread);
}
Copy the code
5 AcquirelNterRuptibly Exclusive interruptible lock acquirer
Before JDK1.5, when a thread fails to acquire the lock and is blocked outside synchronized, if the interrupt operation is performed on the thread, the interrupt flag bit of the thread will be modified, but the thread will still be blocked on synchronized, waiting to acquire the lock, that is, it cannot respond to the interrupt.
Acquire, the exclusive lock acquisition method analyzed above, also does not respond to interrupts. However, AQS provides another acquireInterruptibly template method that returns immediately and throws InterruptedException if the current thread is interrupted while waiting to acquire the lock.
public final void acquireInterruptibly(int arg)
throws InterruptedException {
// If the current thread is interrupted, throw an exception
if (Thread.interrupted())
throw new InterruptedException();
// Try to get the lock
if(! tryAcquire(arg))// If not, call AQS interruptible method
doAcquireInterruptibly(arg);
}
Copy the code
5.1 doAcquireInterruptibly Exclusive interruptible lock acquisition
DoAcquireInterruptibly determines whether the thread is interrupted, and if so, returns and throws an exception. The other steps follow the same basic principle as the exclusive uninterruptible lock. There is also a difference in how the following suspended thread is handled when the thread is interrupted:
- Exclusive uninterruptible lock acquisition simply records the status, interrupted = true, and then the lock acquisition cycle continues.
- An exclusive interruptible lock acquisition throws an exception directly, and therefore jumps directly out of the loop to execute the finally block.
/** * exclusive interruptible lock to get **@param* / arg parameter
private void doAcquireInterruptibly(int arg)
throws InterruptedException {
// Also call addWaiter to add the current thread as a node to the end of the synchronization queue
final Node node = addWaiter(Node.EXCLUSIVE);
// Get lock failure flag, default is true
boolean failed = true;
try {
As with the exclusive uninterruptible acquireQueued method, the lock is acquired in a loop */
for(; ;) {final Node p = node.predecessor();
if (p == head && tryAcquire(arg)) {
setHead(node);
p.next = null; // help GC
failed = false;
return;
}
if (shouldParkAfterFailedAcquire(p, node) &&
parkAndCheckInterrupt())
If a thread is interrupted, only this status is recorded, interrupted = true, Then the loop continues to acquire the lock * * but in this exclusive interruptible lock acquisition method * if the thread is interrupted, an exception is thrown directly here, so the finally block * */ breaks out of the loop directly
throw newInterruptedException(); }}/* The finally block */ is executed when a lock is acquired or an exception is thrown
finally {
/* If the lock fails to be obtained. If the thread is interrupted, cancelAcquire cancels the lock request and the thread terminates
if(failed) cancelAcquire(node); }}Copy the code
5.2 Finally code block
In the doAcquireInterruptibly method, you have a finally block that will execute no matter what happens in the try. In the acquireInterruptibly exclusive method of obtaining the lock, there are only two cases in which finally is executed:
- When the current node (thread) finally acquires the lock, finally is entered and failed = false is set after the lock is acquired.
- If an exception occurs ina try, then skip directly to finally. If an exception occurs in tryAcquire, the predecessor method is more likely to throw InterruptedException because the thread is interrupted. Then enter the finally code block directly. At this point, the lock is not acquired. Failed =true!
- The tryAcquire method is implemented by ourselves, and we decide what exceptions to throw. If an exception is thrown, it is not normally thrown in doAcquireInterruptibly, and may be thrown when tryAcquire is called in the first place.
- The predecessor method checks that nullPointerExceptions are thrown if the predecessor is null. But the comment also says that this check has no code-level meaning, perhaps the exception will never be thrown?
- According to the doAcquireInterruptibly logic, if a thread is interrupted during suspension, an InterruptedException is actively thrown, which is also known as “interruptible.
The logic in the finally code block is:
- If failed = true, finally is performed without obtaining the lock, and an exception occurs. CancelAcquire cancels the current node thread’s request to acquire the lock, doAcquireInterruptibly ends, and raises an exception.
- If failed = false, the lock has been acquired, then nothing is actually executed in finally and the doAcquireInterruptibly method ends.
5.2.1 cancelAcquire Cancels the lock acquisition request
Since the exclusive interruptible lock acquisition method, the thread is interrupted and throws an exception is more common, so here is the analysis of finally cancelAcquire source code. CancelAcquire method is used to cancel the request of node to acquire the lock. The parameter is node to be cancelled. The approximate steps are as follows:
- The thread recorded by node is set to null
- Skip the cancelled front node. Search node forward until it finds a node pred whose state is less than or equal to 0. Update node.prev to the found pred.
- The node waitStatus waitStatus was CANCELLED.
- If node is a tail, try the CAS update tail to pred and continue the CAS set pred. Next to null.
- Otherwise, the node is not a tail or CAS failed (there may be concurrent operations on the tail) :
- If node is not a descendant of head and pred status is SIGNAL or waitStatus of PREd was set to SIGNAL successfully, and the thread of preD record is not NULL. So set pred.next to point to Node.next. Finally, Node. next points to Node itself.
- Otherwise, node is the successor of head or the preD state fails to be set or the thread of preD record is null. Call the unparksucceeded then to wake up an uncancelled successor of the node. Finally, Node. next points to Node itself.
/** * cancel the lock request **@paramNode specifies the node */
private void cancelAcquire(Node node) {
// Ignore if node doesn't exist
if (node == null)
return;
/*1 node sets thread to null*/
node.thread = null;
/ * 2 is similar to shouldParkAfterFailedAcquire method finds the effective precursor code: do {node. Prev = Mr Pred = Mr Pred. Prev. } while (pred.waitStatus > 0); pred.next = node; Here again, node looks forward until it finds a node whose state is less than or equal to zero (that is, the node that has not been canceled), as a precursor, but node.prev is updated, not pred.next*/
Node pred = node.prev;
while (pred.waitStatus > 0)
node.prev = pred = pred.prev;
//predNext Records the successor of preD, which will be used later by CAS.
Node predNext = pred.next;
/*3 nodes' wait status is set to CANCELLED */
node.waitStatus = Node.CANCELLED;
/*4 If the current node is a tail node, then try to update tail to pred and continue CAS setting pred. Next to null. * /
if (node == tail && compareAndSetTail(node, pred)) {
// The next node of pred is set to null, and it does not matter if it fails, indicating that some other thread has been queued or cancelled.
compareAndSetNext(pred, predNext, null);
}
/*5 Otherwise, the node is not a tail or CAS failed (there may be concurrent operations on the tail), in which case all you need to do is to combine pred with node's subsequent non-cancelled nodes. * /
else {
int ws;
/*5.1 If node is not a descendant of head and pred status is SIGNAL or waitStatus of PREd was set to SIGNAL successfully and the thread of preD record is not NULL. So set pred.next to point to Node.next. Prev is not set here, but that's ok. At this point the later turned into a node of the Mr Pred subsequent - next, and the subsequent next node if access to the lock, and then to find the effective precursors in shouldParkAfterFailedAcquire method, can also find this not cancel Mr Pred, at the same time will be next. The prev point to Mr Pred, The prev relationship is set. * /
if(pred ! = head && ((ws = pred.waitStatus) == Node.SIGNAL || (ws <=0&& compareAndSetWaitStatus(pred, ws, Node.SIGNAL))) && pred.thread ! =null) {
// Get the next node
Node next = node.next;
// If the next node exists and is not cancelled
if(next ! =null && next.waitStatus <= 0)
// Then CAS sets perd.next to point to Node.next
compareAndSetNext(pred, predNext, next);
}
/*5.2 Otherwise, node is the successor of head or the preD state fails to be set or the thread of preD record is null. * * The unparksucceeded method needs to be called to try to wake up the successor of the node because node, being the successor of the head, is the only one that is eligible to try to get the lock. * If the external thread A releases the lock but has not called unpark to wake up the node, then the node is interrupted or an exception occurs, then the node will call cancelAcquire to cancel, and the record thread inside the node becomes null. * At this time, the unpark method of thread A is executed. Locksupport. unpark(null) will not wake up any nodes * then nodes behind the node will not wake up, and the queue will be deactivated; If, in this case, a call to the unparksucceed * in the code that Node would have called cancelAcquire cancel would wake up the successors of the cancelled nodes so that they could try to acquire the lock, thereby guaranteeing queue activity! * * The node is not completely removed from the queue, * the awakened node will attempt to acquire the lock, and after obtaining the lock, the * setHead(node); * p.next = null; // help GC * section, which may clear these cancelled nodes * */
else {
unparkSuccessor(node);
}
/* Finally, node.next refers to node itself, which can be used to destroy invalid nodes directly during GC and also for Condition's isOnSyncQueue method to determine whether a node that was previously in the conditional queue has been moved to the synchronous queue. Since the next field of a node is used in the synchronization queue, if the next field of a node has a value, we can assert that the node with a value in the next field must be on the synchronization queue. It can also be seen that the reverse order traversal should be used when traversal, otherwise the use of positive order traversal may appear dead loop */node.next = node; }}Copy the code
5.2.2 cancelAcquire case demonstration
Assume a synchronous queue structure where ABCDE five threads call the acquireInterruptibly method to claim the lock, and BCDE threads are all blocked because they cannot obtain the lock.
Let’s look at how cancelAcquire works in several cases:
If thread D is interrupted at this point, then an exception is thrown into the finally block, which belongs to node but is not a tail, and node is not a successor to head, as shown below:
The structure after the cancelAcquire method is as follows:
If thread E is interrupted, then an exception is thrown into the finally block, which is the case when Node is the last node, as shown below:
The structure after the cancelAcquire method is as follows: If two new threads F and G are added and both threads are suspended, then the structure of the synchronization queue is shown below: As you can see, the queue actually bifurcated, which is quite common in synchronous queues, because the cancelled node did not actively remove its own prev reference. So this part of the cancelled node can not be deleted? In fact, it is possible, but it needs to meet certain conditions of the structure! If thread B is interrupted at this point, then an exception is thrown into the finally block, which belongs to node, not the tail, and node is the successor of head, as shown below: The structure after the cancelAcquire method is as follows:
Note that in this case Node would also call the unparksucceeded method to wake up the successor C and let it try to get the lock, and if thread A had not run out of locks at that time, then C certainly could not get the lock at that time.
C does not do nothing, however. During the time C gains CPU execution after being awakened, it changes some references in the doAcquireInterruptibly method’s for loop.
If C is suspended, waitStatus=1, waitStatus=1, waitStatus=1, waitStatus=1, waitStatus=1, waitStatus=1, waitStatus=1, waitStatus=1, waitStatus=1, waitStatus=1, waitStatus=1
As you can see, C is eventually linked directly to the head node, but B is not GC yet because it has a prev reference. Don’t worry, this is because it has not reached the specified structure, so it will be GC when it arrives.
If thread A runs out of resources at this point, the lock is first released, and an attempt is made to wake up A successor thread that has not been canceled, obviously choosing C.
If C is woken up after A releases the lock and before C is called locksupport. unpark, C is woken up by A step interrupt. When C throws an exception, it will not acquire the lock any more, but will finally execute cancelAcquire method. At this time, it still belongs to node, not the tail node, node is the successor of head, as shown in the following figure:
Then after C executes cancelAcquire, the structure is as follows:
Unpark locksupport. unpark(NULL) if thread A has acquired CPU execution, locksupport. unpark(null) will do nothing. Wouldn’t the queue then be inactivated? Not really!
The unparkprecursor method of the cancelAcquire method is now crucial in the case of “node is not a trailing node, node is the successor of head”. This method is used to wake up an uncancelled successor of C, F, to attempt to acquire the lock, thus ensuring that the queue is alive.
After F is woken up, it will judge whether it can sleep, obviously not, because the state of the precursor node is 1. At this point, after a series of operations in the loop, it will become the following structure:
Obviously F is the direct successor of head and can acquire the lock.
After acquiring the lock, F will set itself as the new head, which will change some of the references, that is, F will remove the prev and next relationships with the precursor:
setHead(node);
p.next = null; // help GC
Copy the code
Under the structure after the reference relationship changes:
As you can see, it is at this point that the invalid nodes are actually removed and collected by the GC. So, what does it take to actually delete a node? The condition is: if a node has acquired the lock, then the node’s precursor and the node related to the node’s precursor will be deleted!
However, in the above analysis, we assume that as long as there is a node reference association, it will not be collected by GC, whereas in reality modern Java virtual machines use a reachable analysis algorithm to analyze garbage, so in the above queue, for those “forks”, That is, nodes that are cancelled, only prev references are left, and most importantly cannot be reachable through the head and tail reference chains and have no external references are marked as garbage in the reachable analysis algorithm and collected directly in a GC!
For example, when thread F completes execution and G is the next thread, F will be deleted after G obtains the lock. The final structure is as follows:
6 tryAcquireNanos Exclusive timeout obtains the lock
The tryAcquireNanos template method can be thought of as an “enhanced version” of the interrupt acquireInterruptibly method.
/** * exclusive timeout lock, support interrupt **@paramArg parameter *@paramNanosTimeout Timeout, nanoseconds *@returnCheck whether the lock is successfully obtained *@throwsInterruptedException Throws an InterruptedException */ if interrupted
public final boolean tryAcquireNanos(int arg, long nanosTimeout)
throws InterruptedException {
// If the current thread is interrupted, throw an exception
if (Thread.interrupted())
throw new InterruptedException();
// tryAcquire is also called to attempt to acquire the lock. If it succeeds, return true
Otherwise, the doAcquireNanos method is suspended for a specified period of time. This is true if the lock has been acquired for a short time, and false if the lock has not been acquired for a long time
return tryAcquire(arg) ||
doAcquireNanos(arg, nanosTimeout);
}
Copy the code
6.1 doAcquireNanos Obtains the lock in exclusive timeout mode
The doAcquireNanos(int ARg, Long nanosTimeout) method adds a timeout fetch feature to support response interrupts.
In this method, in the spin process, when the node’s precursor node is the head node, it tries to acquire the lock, and if it succeeds in obtaining the lock, it returns from this method. This process is similar to the process of exclusive synchronous acquisition, but there are differences in the processing of lock acquisition failure.
If the current thread fails to acquire the lock, determine whether it has timed out (nanosTimeout is less than or equal to 0, indicating that it has timed out), if not, recalculate the timeout interval nanosTimeout, and then make the current thread wait nanosTimeout nanoseconds (when the set timeout has been reached, The thread returns from locksupport. parkNanos(Objectblocker,long Nanos).
If nanosTimeout is less than or equal to the spinForTimeoutThreshold (1000 nanoseconds), the thread is not made to wait out the timeout, but to enter a fast spin process. The reason is that very short timeout waits cannot be very precise, and if timeout waits are made at this point, nanosTimeout timeouts as a whole appear imprecise.
Therefore, in the case of very short timeouts, the AQS will go into unconditional fast spin instead of suspending the thread.
static final long spinForTimeoutThreshold = 1000L;
/** * exclusive timeout to get lock **@paramArg parameter *@paramNanosTimeout Remaining timeout, nanoseconds *@returnTrue on success; * or false on failure@throwsInterruptedException Throws an InterruptedException */ if interrupted
private boolean doAcquireNanos(int arg, long nanosTimeout)
throws InterruptedException {
// Get the current nanosecond time
long lastTime = System.nanoTime();
// Also call addWaiter to add the current thread as a node to the end of the synchronization queue
final Node node = addWaiter(Node.EXCLUSIVE);
boolean failed = true;
try {
As with the exclusive uninterruptible acquireQueued method, the lock is acquired in a loop */
for(; ;) {final Node p = node.predecessor();
if (p == head && tryAcquire(arg)) {
setHead(node);
p.next = null; // help GC
failed = false;
return true;
}
/* This is the difference */
// If the remaining timeout is less than 0, exit the loop and return false, indicating that the lock was not acquired
if (nanosTimeout <= 0)
return false;
// If suspend is required and the remaining nanosTimeout is greater than spinForTimeoutThreshold, i.e., greater than 1000 nanoseconds
if (shouldParkAfterFailedAcquire(p, node)
&& nanosTimeout > spinForTimeoutThreshold)
// Then call the locksupport. parkNanos method to suspend nanosTimeout from the current thread
LockSupport.parkNanos(this, nanosTimeout);
// Get the current nanosecond, which is probably due to the thread being woken up in mid-stream
long now = System.nanoTime();
// Calculate new remaining timeout: old remaining timeout - (current time now - last calculated time lastTime)
nanosTimeout -= now - lastTime;
//lastIme is assigned to the time of this calculation
lastTime = now;
// If the thread is interrupted, throw an exception
if (Thread.interrupted())
throw newInterruptedException(); }}/* The finally block */ is executed when a lock is acquired, a timeout expires, and an exception is thrown
finally {
/* If the lock fails to be obtained. CancelAcquire () : cancelAcquire () : cancelAcquire () : false * */
if(failed) cancelAcquire(node); }}Copy the code
6.2 Finally code block
In the doAcquireNanos method, you have a finally block that will execute no matter what happens in the try. In the tryAcquireNanos exclusive timeout lock method, there are only three cases where finally is executed:
- When the current node (thread) finally acquires the lock, finally is entered and failed = false is set after the lock is acquired.
- If an exception occurs ina try, then skip directly to finally. If an exception occurs in tryAcquire, the predecessor method is more likely to throw InterruptedException because the thread is interrupted. Then enter the finally code block directly. At this point, the lock is not acquired. Failed =true!
- The tryAcquire method is implemented by ourselves and is up to us to decide what exceptions are thrown. Even if an exception is thrown, it will not normally be thrown in doAcquireNanos and will probably be thrown when tryAcquire is first called.
- The predecessor method checks that nullPointerExceptions are thrown if the predecessor is null. But the comment also says that this check has no code-level meaning, perhaps the exception will never be thrown?
- According to doAcquireNanos logic, if a thread is interrupted during suspension, InterruptedException is actively thrown, which is also known as “interruptible.”
- Method timeout, the current thread has not acquired the lock, then will break out of the loop, directly into the finally code block, has not acquired the lock, failed=true!
The logic in the finally code block is:
- If failed = true, finally is performed without obtaining the lock. An exception may occur or the timeout time is up. CancelAcquire cancels the lock request of the current thread, doAcquireNanos terminates, raises an exception or returns false.
- If failed = false, the lock has been acquired, then nothing is actually executed in finally, and the doAcquireNanos method ends, returning true.
Summary of exclusive lock acquisition/release
To acquire and release locks exclusively, we need to override the tryAcquire and tryRelease methods.
Exclusive lock acquisition and lock release require the tryAcquire method to record which thread acquired the lock. The ‘exclusiveOwnerThread’ field (setExclusiveOwnerThread) is generally used. The value of this field is known after tryRelease.
7.1 Acquire/Release Flow chart
Acquire process:
The release process:
7.2 Acquire General Process
Based on the above source code, we try to summarize the general process of acquire method (exclusive acquire lock) to build synchronous queue as.
Thread A has acquired the lock in acquire’s tryAcquire method, and the synchronization queue has not been initialized. Head and tail are null.
When the second thread B enters the queue, since A has acquired the lock, the thread will be constructed as A node and added to the queue. In the enQ method, the first loop, since tail is null, will construct A null node as the head and tail of the synchronous queue:
On the second loop, the node will be added to the end of the node, with tail pointing to it!
Then in acquireQueued approach, assuming that node spin lock is not obtained, then the shouldParkAfterFailedAcquire method will set the precursor node waitStatus = 1, then the node’s thread B will be suspended:
Next, if thread C also attempts to acquire the lock, assuming it does not acquire it, then C will also be suspended:
The SIGNAL state (-1) of a node is set by its stepchild (-1).
Here acquire general process analysis completed!
7.3 General process of Release
Based on the above source code, we try to summarize the general flow of release method (exclusive lock release) as follows:
If thread A has succeeded in using the shared resources, it calls unlock and internal release, then calls tryRelease first to release the lock, then calls the unparkprecursor method and sets the state of the head node to 0. And wakes up the uncancelled successor of the head node (waitStatus not greater than 0), which is clearly a THREAD B node. This is the end of the resize method, and here is what happens to the awakened node.
After calling unpark to wake up thread B, thread B continues execution in parkAndCheckInterrupt, first checking the interrupt status and recording the reason for being woken up. In this case, thread B was not woken up because of the interrupt, so returning false, AcquireQueued has the interrupted field set to false.
Thread B then continues to spin in the acquireQueued method and, assuming that thread B has acquired the lock, calls the setHead method to clear the thread record and sets node B as the head. It doesn’t matter that the thread records inside the node are cleared, because in our tryAcquire method we would normally record which thread acquired the lock.
When the last blocking node is awakened and thread E has acquired the lock, the structure of the synchronization queue is as follows:
When the last thread E calls unlock after using the shared resource, and after releasing the lock in release, tries to wake up the succeeding nodes using head, it can be determined that the waitStatus of the head node is still equal to 0, so the unparkprecursor method will not be invoked.
The general process analysis is finished here release!
If you don’t understand or need to communicate, you can leave a message. In addition, I hope to like, collect, pay attention to, I will continue to update a variety of Java learning blog!