A list,
A ReentrantLock is a ReentrantLock.
Reentrant: The same thread can acquire the same lock more than once. As shown below, method1 and method2 use the same lock, and when method2 is called in method1, the calling thread can repeatedly acquire the lock; Otherwise, the thread will never get a “second lock” to execute method2.
Synchronized is also reentrant.
ReentrantLock lock = new ReentrantLock();
public void method1() {
lock.lock();
// do some , call method2
method2();
lock.lock();
}
public void method2(){
lock.lock();
// do some
lock.unlock();
}
Copy the code
Two, principle analysis
- Concurrent locking and releasing is done through the internal class Sync
- Sync inherits AQS (Abstract Queue synchronizer)
- CAS
Classification:
- Fair and unfair locks
- Exclusive locks and shared locks
Sync has two implementation classes: FairSync and NonfairSync. By default, NonfairSync non-fair locks are created.
Principle:
In AQS, the variable state that is volatile is maintained, indicating that state is visible in memory. When a thread changes the value, it is flushed back to main memory to ensure that the variable seen by each thread is up to date.
/**
* The synchronization state.
*/
private volatile int state;
Copy the code
- NonfairSync
/** * Performs lock. Try immediate barge, Backing up to normal * acquire on failure. */ final void lock() {if (compareAndSetState(0, 1)) setExclusiveOwnerThread(Thread.currentThread()); else acquire(1); }Copy the code
An unfair lock will immediately acquire the lock. If it cannot acquire the lock, it will enter the block to acquire the lock, that is, it will have a chance to acquire the lock at first.
Next, acquire(1) :
Public final void acquire(int arg) {// If (! TryAcquire (arG) && // Join queue acquireQueued(addWaiter(Node.exclusive), arg) selfInterrupt(); }Copy the code
- TryAcquire (ARG) This is the core method, concrete implementation:
final boolean nonfairTryAcquire(int acquires) { final Thread current = Thread.currentThread(); int c = getState(); If (c ==0) {// c==0 indicates that no thread is currently holding the lock. if (compareAndSetState(0, acquires)) { setExclusiveOwnerThread(current); return true; }} // the state value is increased by 1, indicating that the thread is holding the lock. Else if (current == getExclusiveOwnerThread()) {// acquires parameter 1 int nexTC = c + acquires; if (nextc < 0) // overflow throw new Error("Maximum lock count exceeded"); // setState to the increased value setState(nextc); return true; } return false; }Copy the code
- If tryAcquire(ARG) returns false and no lock was acquired, it is enqueued until the lock is acquired or interrupted.
Private Node addWaiter(Node mode) {Node Node = new Node(thread.currentThread (), mode); Node pred = tail; if (pred ! = null) { node.prev = pred; // The CAS atomic operation is also used to prevent the loss of some nodes caused by multiple threads adding nodes at the same time. if (compareAndSetTail(pred, node)) { pred.next = node; return node; }} // If pred == null, i.e. there are no elements in the queue, enq(node) is initialized; return node; }Copy the code
Initialization process:
private Node enq(final Node node) { for (;;) { Node t = tail; If (t == null) {// Must initialize // create a Node as head, CAS if (compareAndSetHead(new Node()))) // tail = head; } else { node.prev = t; // CAS, this step is taken whenever a thread has successfully created a head or tail node. if (compareAndSetTail(t, node)) { t.next = node; return t; }}}}Copy the code
The first loop creates the head, and the tail node is the same as the head node; The second loop will execute else logic to add node to tail node, which will become the latest tail node. If there are other concurrent threads executing the initialization method, else code will be executed, since CAS is also used to initialize the head and tail nodes and add new nodes.
Then add the node to the queue:
final boolean acquireQueued(final Node node, int arg) { boolean failed = true; try { boolean interrupted = false; // Loop for (;;) { final Node p = node.predecessor(); If (p == head && tryAcquire(arg)) {setHead(node); if (p == head && tryAcquire(arg)) {setHead(node); p.next = null; // help GC failed = false; return interrupted; } / / after a failed every time whether to need to hang the if (shouldParkAfterFailedAcquire (p, node) && parkAndCheckInterrupt ()) / / both method returns true, said thread needs to hang, And the thread interrupt flag is true. // This can be used to determine whether the thread has experienced an interrupt until it recovers from the wait state. interrupted = true; } } finally { if (failed) cancelAcquire(node); }}Copy the code
The thread blocks here and continues to acquire the lock (becoming eligible to compete for it as a second node) until the lock is acquired or interrupted, returning whether the thread was interrupted during the suspension process.
Look at shouldParkAfterFailedAcquire (), parkAndCheckInterrupt () :
// private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) { int ws = pred.waitStatus; If (ws == node. SIGNAL) return true; '//' CANCELLED 'means CANCELLED. At this point, the current node should be searched for the first non-cancelled node and become the subsequent node, because it has no meaning if the precursor node of a node is CANCELLED. if (ws > 0) { do { node.prev = pred = pred.prev; } while (pred.waitStatus > 0); pred.next = node; } else { compareAndSetWaitStatus(pred, ws, Node.SIGNAL); } return false; } // private final Boolean parkAndCheckInterrupt() {// Locksupport.park (this); // Return thread.interrupted (); }Copy the code
WaitStatus is defined by Node. It identifies the wait state of the thread. It has four values:
-
CANCELLED = 1: Thread has been CANCELLED
-
SIGNAL = -1: Subsequent threads of the current thread need to be unpark.
-
CONDITION = -2: The thread (in CONDITION sleep) is waiting for the CONDITION to wake up
-
PROPAGATE = — 3:(PROPAGATE for shared lock)
If a node’s status is SIGNAL, its next node needs to be woken up.
Let’s go back to this method acquire() :
Public final void acquire(int arg) {// If (! TryAcquire (arG) && // Join queue acquireQueued(addWaiter(Node.exclusive), arg) selfInterrupt(); }Copy the code
AcquireQueued (addWaiter(Node.exclusive), arg) returns if the thread is interrupted, selfInterrupt() is executed; Thread interrupts need to be studied.
-
FairSync
The process of fair lock is basically the same as that of unfair lock, but the difference lies in:
// FairSync protected final boolean tryAcquire(int acquires) { final Thread current = Thread.currentThread(); int c = getState(); If (c == 0) {// If (c == 0) { hasQueuedPredecessors() && compareAndSetState(0, acquires)) { setExclusiveOwnerThread(current); return true; } } else if (current == getExclusiveOwnerThread()) { int nextc = c + acquires; if (nextc < 0) throw new Error("Maximum lock count exceeded"); setState(nextc); return true; } return false; }Copy the code
The difference between an unfair lock and a fair lock is shown in the conditional code block if(c==0) :
/ / -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- if not fair lock (c = = 0) {if (compareAndSetState (0, acquires)) {setExclusiveOwnerThread (current); return true; }} / / -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- fair lock -- -- -- -- -- the if (c = = 0) {/ / if (! hasQueuedPredecessors() && compareAndSetState(0, acquires)) { setExclusiveOwnerThread(current); return true; }}Copy the code
This is the locking process.
Lock release process:
The lock is actually released by calling the release() method:
public final boolean release(int arg) { if (tryRelease(arg)) { Node h = head; if (h ! = null && h.waitStatus ! = 0) unparkSuccessor(h); return true; } return false; }Copy the code
TryRelease () : Releases the lock
Protected final Boolean tryRelease(int releases) {// Change the state value, usually minus 1 int c = getState() -releases; if (Thread.currentThread() ! = getExclusiveOwnerThread()) throw new IllegalMonitorStateException(); boolean free = false; // In case of reentrant, the same thread has acquired the lock more than once, and the current thread can release the lock only if state is reduced to 0. if (c == 0) { free = true; setExclusiveOwnerThread(null); } setState(c); return free; }Copy the code
Unparkprecursor () : Wake up the successor node
Private void unparksucceeded (Node Node) {// The state of the current Node changed to 0 int ws = node.waitStatus; if (ws < 0) compareAndSetWaitStatus(node, ws, 0); // If the subsequent node is null or cancelled, there is no point in looking back until the first valid node. Node s = node.next; 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; } // Wake up valid nodes found. Previously, the thread failed to acquire the lock and then blocked. if (s ! = null) LockSupport.unpark(s.thread); }Copy the code
-
Lock(), tryLock(), tryLock(long timeout, TimeUnit unit)
- Lock () will keep fetching the lock until it returns
- TryLock () means to try to obtain the lock and return immediately if not
- TryLock (long timeout, TimeUnit Unit) Attempts to obtain the lock. If the lock has not been obtained after the timeout, false is returned
-
The lock () and lockInterruptibly ()
Both methods suspend the thread while acquiring the lock, but the thread will eventually recover from the suspended state at some point, usually for two reasons:
- Woken up by another thread. For example, the task in the second position in the queue is called when the first task is finished
unparkSuccessor(h)
Notify the thread to acquire the lock. - Interrupted by another thread.
The difference between lock() and lockInterruptibly() is that when a thread is interrupted by another thread, lock() proceeds to acquire the lock, records the interruption, and finally sets the interrupt flag by calling selfInterrupt() once the lock is acquired. (Whether status is reset depends on the case, Refer to the source code notes); LockInterruptibly () does not continue to acquire the lock and throws InterruptedException. The exception can be caught by our program and its handling is up to the caller.
- Woken up by another thread. For example, the task in the second position in the queue is called when the first task is finished
/ / lock () if (shouldParkAfterFailedAcquire (p, node) && parkAndCheckInterrupt ()) / / just did mark, do not affect process interrupted = true; ------------------------------------------------- // lockInterruptibly() if (shouldParkAfterFailedAcquire(p, Node) && parkAndCheckInterrupt()) // Throw new InterruptedException();Copy the code
So much for ReentrantLock lock and release source code analysis.Copy the code
Third, summary
Here is a basic flow chart of unfair locks:
This article is formatted using MDNICE