In the last article we introduced the source code of AQS data structure and implementation principle, in today’s article we introduce the implementation principle of ReentrantLock, also look at how to use AQS in ReentrantLock.
The content of this article, depends on the previous article, we need to understand the source code of AQS, the article is connected to: AQS source code
ReentrantLock is a Java level lock implementation and a common lock object. It is a reentrant exclusive lock.
1 class structure
ReentrantLock is an implementation class of Lock that implements the functionality that locks defined in Lock should have. Its class diagram is as follows:
In ReentrantLock, there is only one variable Sync of type Sync. The lock function is implemented by changing the class and its subclasses.
This class inherits from AQS and has two implementation classes, FairSync and NonfairSync, which are fair and unfair implementations of ReentrantLock, respectively.
2 the lock and unlock
The main function of a lock is to lock and release the lock. In this section we will understand the implementation principle according to the source code.
The methods in reentrantLock. Sync are as follows:
In the previous article, we learned that some AQS methods require subclasses. In the figure above, we can see that the methods tryAcquire and tryRelease are implemented.
Let’s take a look at how fair and unfair locks are handled in ReentrantLock.
2.1 Obtaining an Unfair Lock
NonfairSync lock method source code:
final void lock(a) {
// Change the state value by performing the CAS operation
if (compareAndSetState(0.1))
// Lock preemption succeeded in setting the thread holding the exclusive lock as the current thread
setExclusiveOwnerThread(Thread.currentThread());
else
The tryAcquire method is called in the AQS method
acquire(1);
}
protected final boolean tryAcquire(int acquires) {
return nonfairTryAcquire(acquires);
}
final boolean nonfairTryAcquire(int acquires) {
// Get the current thread
final Thread current = Thread.currentThread();
// Get the state value
int c = getState();
// state: 0 indicates no lock
if (c == 0) {
// CAS operation replaces state
if (compareAndSetState(0, acquires)) {
setExclusiveOwnerThread(current);
return true; }}// The current thread has acquired the lock representing reentrant
else if (current == getExclusiveOwnerThread()) {
// Add state
int nextc = c + acquires;
if (nextc < 0) // overflow
throw new Error("Maximum lock count exceeded");
// Change the state value. No CAS operation is required
setState(nextc);
return true;
}
return false;
}
Copy the code
The unfair lock acquisition logic of ReentrantLock is relatively simple. When the lock method is called, the lock preemption operation is performed first. If the preemption fails, the lock is added to the blocking queue.
2.2 Obtaining fair Locks
FairSync lock method:
final void lock(a) {
// Call acquire directly in AQS
acquire(1);
}
protected final boolean tryAcquire(int acquires) {
// The current thread
final Thread current = Thread.currentThread();
// Current status value
int c = getState();
// State 0 indicates no lock
if (c == 0) {
// Perform lock preemption when there are no other waiting threads
if(! hasQueuedPredecessors() && compareAndSetState(0, acquires)) {
setExclusiveOwnerThread(current);
return true; }}// The reentrant logic is consistent with the logic of unfair locking
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 lock logic of fair lock is also relatively simple. When there are no other waiting threads in the blocking queue, the lock is preempted, which is different from the unfair lock.
2.3 Lock Release
The tryRelease method in reentrantLock. Sync is as follows:
protected final boolean tryRelease(int releases) {
// State minus the number of locks to be released
int c = getState() - releases;
// An exception was thrown by the thread that did not acquire the exclusive lock
if(Thread.currentThread() ! = getExclusiveOwnerThread())throw new IllegalMonitorStateException();
// Whether to release all
boolean free = false;
if (c == 0) {
free = true;
// Set the thread owning the exclusive lock to NULL
setExclusiveOwnerThread(null);
}
// Change the state value
setState(c);
// The AQS will wake up the next node in the blocking queue
return free;
}
Copy the code
So far ReentrantLock in the lock acquisition and release logic, we will explain the end, the logic is relatively simple, we suggest that when looking at the source code to point to AQS look, deepen the understanding of AQS.
The lock status of a ReentrantLock is marked by the state property, which has the following values:
- 0 no lock
- The number of times a thread greater than 0 acquires the lock is added to the reentrant value
When the lock is acquired and released, the state is added and subtracted. When it reaches 0, the lock is released successfully and the next node in the AQS blocking queue is awakened to compete for the lock.
When there are other blocking threads in the blocking queue, the fair lock will be inserted into the end of the blocking queue, while the non-fair lock will be locked once during the adding operation.
3 summary
In today’s article, I introduced how ReentrantLock is implemented on the basis of AQS lock function, no complicated logic. We can combine with the AQS source code to have a look.
This is the end of today’s article. The principle of CountDownLatch synchronization tool will be introduced in the next article. You can take a look at it first, which is also implemented through AQS and the logic is not difficult.
If you feel helpful, welcome to pay attention to the public number, your attention is my biggest motivation to update ~