The lock

In the case of locks in Java, it is also possible for a lock to occupy multiple criteria and meet multiple classifications, such as a ReentrantLock that is both interruptible and reentrant.

According to the classification standards, we divide locks into the following 7 categories, which are:

  • Biased/lightweight/heavyweight locks;

  • Reentrant lock/non-reentrant lock;

  • Shared/exclusive lock;

  • Fair/unfair lock;

  • Pessimistic lock/optimistic lock;

  • Spin-lock/non-spin-lock;

  • Interruptible locks/non-interruptible locks

Bias lock/lightweight lock/heavyweight lock

The first category is biased lock/lightweight lock/heavyweight lock. These three types of locks specifically refer to the synchronized lock state, which is indicated by the mark word in the object header.

  • Biased locking

If there’s no competition for the lock all the time, then there’s no need to lock it, just mark it, and that’s the idea of favoring locking. After an object is initialized, there is no thread to obtain its lock, it is can be biased, when the first thread to access it and try to get the lock, it will record this thread, if you try to get the lock after the thread is the owner of biased locking, can directly obtain locks, overhead is very small, the best performance.

  • Lightweight lock

JVM developers have found that in many cases synchronized code is executed alternately by multiple threads rather than at the same time, meaning that there is no actual contention or only a short lock contention that CAS can resolve, in which case a fully mutually exclusive heavyweight lock is unnecessary. A lightweight lock is one that is accessed by another thread when the lock is biased, indicating that there is a race. Then the biased lock will be upgraded to a lightweight lock, and the thread will try to acquire the lock by spinning instead of blocking.

  • Heavyweight lock

Heavyweight locks are mutex locks, which are implemented using the synchronization mechanism of the operating system, so the overhead is relatively high. When multiple threads compete with each other for a long time, the lightweight lock cannot meet the demand, and the lock will expand to the heavyweight lock. Heavyweight locks block other threads that apply but can’t get the lock.

In summary, biased locking performs best and avoids CAS operations. Lightweight locks use spin and CAS to avoid thread blocking and wake up with moderate performance. Heavyweight locks block threads that do not acquire the lock and perform worst

Reentrant lock/non-reentrant lock

Shared/exclusive lock

A shared lock means that the same lock can be acquired by multiple threads at the same time, whereas an exclusive lock means that the lock can only be acquired by one thread at the same time. Our read-write locks best illustrate the concept of shared and exclusive locks. Read locks are shared locks, while write locks are exclusive locks. Read locks can be simultaneously read and held by multiple threads, while write locks can be held by at most one thread at a time

Fair lock/unfair lock

The fair meaning of fair lock is that if the thread can not get the lock now, then all the threads will enter the wait and queue. The thread that waits for a long time in the wait queue will get the lock first, which means first come, first served. But non-fair lock is not so “perfect”, it will under certain circumstances, ignore the thread already in the queue, queue jumping phenomenon

Pessimistic lock/optimistic lock

The concept of pessimistic lock is that before acquiring a resource, the lock must be acquired first, so as to achieve the “exclusive” state. When the current thread is manipulating the resource, other threads cannot obtain the lock, so other threads cannot affect me. Optimistic locking, on the other hand, does not require the lock to be taken before acquiring the resource, nor does it lock the resource; In contrast, optimistic locking takes advantage of the CAS concept to modify a resource without monopolizing it.

Spinlocks/non-spinlocks

The idea behind spin locking is that if a thread can’t get a lock now, instead of blocking or releasing CPU resources, it starts using a loop, which is metaphorically described as “spinning”, as if the thread is “spinning on itself”. In contrast, the idea of non-spin locking is that there is no spin in the process, and if you can’t get the lock you simply give it up, or do some other processing logic, such as queuing, blocking, etc

Interruptible locks/non-interruptible locks

In Java, the synchronized keyword represents an unbreakable lock. Once a thread has claimed the lock, there is no going back until it has acquired it before it can perform any other logical processing. Our ReentrantLock is a typical interruptible lock. For example, if you use the lockInterruptibly method to acquire a lock and suddenly don’t want to acquire it, you can do something else after the interrupt without waiting until the lock is acquired

Pessimistic lock/optimistic lock nature

Pessimistic locking more pessimistic, it thinks if don’t lock the resources, other threads will be up for, will cause data errors, so pessimistic locks in order to ensure the correctness of the result, in every time access and modify data, all the data lock, for other threads to access the data, so we can ensure that the data content

The “Monitor lock” behind Synchronized

The timing of the monitor lock acquisition and release

The simplest way to synchronize code is to use the synchronized keyword to modify a block of code or a method, so that only one thread of protected code can run at a time. Synchronized is implemented using the monitor lock. Each Java object can be used as a lock to implement synchronization. This lock is also known as a built-in lock or monitor lock. The only way to obtain a monitor lock is to enter the synchronized code block or synchronized method protected by the lock. A thread automatically acquires the lock before entering a block of synchronized protected code, and automatically releases the lock when it exits, either through a normal path or by throwing an exception.

Choose between synchronized and Lock.

The same

Synchronized and Lock have a lot in common, and we’ll focus on three big similarities here.

1. Synchronized and Lock are used to protect resource threads.

2. Visibility is guaranteed.

For synchronized, thread A operates before or within A synchronized block, which is visible to subsequent thread B that acquires the same monitor lock. Thread B can see the previous actions of thread A, which also reflects A principle of happens-before against synchronized

For Lock, it is the same as synchronized, and can ensure visibility. As shown in the figure, all operations before unlocking are visible to all operations after locking3. Synchronized and ReentrantLock both have reentrant features

ReentrantLock is one of the main implementation classes of the Lock interface. When comparing synchronized and Lock, the main implementation class of Lock will be selected for comparison. Reentrant means that a thread that has acquired a lock and now attempts to request it again is reentrant if it does not have to release the lock early but can simply continue to use the lock it holds. If the lock must be released before it can be reapplied for, it is not reentrant. Synchronized and ReentrantLock both have reentrant properties

The difference between

  • Add unlock order is different

For Lock, if there are multiple Lock locks, the Lock can be unlocked in reverse order. For example, we can obtain the Lock1 Lock first, and then the Lock2 Lock. When unlocking, the Lock1 Lock is unlocked first, then the Lock2 Lock is unlocked.

lock1.lock(); lock2.lock(); . lock1.unlock(); lock2.unlock();Copy the code

Synchronized, however, cannot do this. The order in which synchronized unlocked and locked must be reversed, for example:


synchronized(obj1){

    synchronized(obj2){ ... }}Copy the code

So in this case, the sequence is first lock obj1, then lock obj2, then unlock obj2, and finally unlock obj1. This is because synchronized is implemented by the JVM and will be unlocked automatically after the execution of synchronized blocks. Therefore, synchronized will be unlocked in the nested order and cannot be controlled by itself.

  • Synchronized locks are not flexible enough

Once a synchronized lock has been acquired by a thread, it can only be blocked by other threads trying to acquire it until the thread that holds the lock finishes running or an exception occurs to release the lock. If the thread holding the lock holds it for a long time before releasing it, the entire program becomes less efficient, and if the thread holding the lock never releases it, the thread trying to acquire the lock will have to wait forever.

The Lock class, by contrast, uses the lockInterruptibly method while waiting for a Lock, which gives it the flexibility to interrupt and exit if the wait becomes too long, to try to obtain the Lock with a tryLock() method, or to do something else if the Lock is not available.

  • Synchronized locks can only be owned by one thread at a time, but Lock locks do not have this limitation

Read locks, for example, can be held by multiple threads at the same time, whereas synchronized cannot.

How to choose

Synchronized and Lock in Java concurrency programming practice and Java core technology both agree that:

Use neither Lock nor synchronized if you can. Because in many cases you can use the mechanism in the java.util.concurrent package, which handles all lock and unlock operations for you, it is recommended to use utility classes in preference for unlocking.

If the synchronized keyword is appropriate for your program, use it to reduce the amount of code you need to write and the chances of errors. Because unlock can go badly wrong if you forget to use it in finally, and synchronized is safer.

Use Lock only if you need special features of Lock, such as try to acquire locks, interruptibility, timeout, etc

Lock common methods


public interface Lock {

    void lock(a);

    void lockInterruptibly(a) throws InterruptedException;

    boolean tryLock(a);

    boolean tryLock(long time, TimeUnit unit) throws InterruptedException;

    void unlock(a);

    Condition newCondition(a);

}

Copy the code

The five methods are lock(), tryLock(), tryLock(Long time, TimeUnit Unit), lockInterruptibly(), unlock().

The lock () method

The Lock interface declares four methods to obtain locks (Lock(), tryLock(), tryLock(long time, TimeUnit Unit), and lockInterruptibly()). What are the differences between the four methods?

First, lock() is the most basic way to acquire locks. The simplest way to acquire a lock is to wait if the lock has already been acquired by another thread.

Lock acquisition and Lock release are both explicit to the Lock interface, unlike synchronized, which is implicit, so Lock does not automatically release the Lock when an exception occurs (synchronized can release the Lock even without writing the corresponding code). Locking and releasing locks must be written in code, so when using a lock(), we must release the lock ourselves. Therefore, it is best practice to first operate on the synchronized resource in a try{} after executing the lock() and catch exceptions if necessary. Then release the lock in finally{} to ensure that the lock is released when an exception occurs, as shown in the sample code below.

Fair and Unfair Locks Why are unfair locks needed?

Compare the advantages and disadvantages of fair and unfair

Equality fair lock the advantage of individual threads, each thread waits for a period of time, all have the chance to perform, and its shortcoming is that the overall execution speed is more slow, throughput, contrary to the fair lock’s advantage lies in the overall execution speed faster, more throughput, but also may produce thread starvation problem, that is to say if the thread has been cut in line, Then the thread in the wait queue may not run for a long time.

  • Source code analysis

In already class contains a Sync class, this class inherits from AQS (AbstractQueuedSynchronizer), the code is as follows:


public class ReentrantLock implements Lock.java.io.Serializable {

 

private static final long serialVersionUID = 7373984872572414699L;

 

/** Synchronizer providing all implementation mechanics */

 

private final Sync sync;

Copy the code

Sync class code:


abstract static class Sync extends AbstractQueuedSynchronizer {... }Copy the code

Sync has two subclasses: FairSync and NonfairSync.


static final class NonfairSync extends Sync {... }static final class FairSync extends Sync {... }Copy the code

Fair lock and non – fair lock lock method source


protected final boolean tryAcquire(int acquires) {

    final Thread current = Thread.currentThread();

    int c = getState();

    if (c == 0) {

        if(! hasQueuedPredecessors() &&Hasqueuedtoraise ()

                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

Unfair lock lock source code is as follows:


final boolean nonfairTryAcquire(int acquires) {

    final Thread current = Thread.currentThread();

    int c = getState();

    if (c == 0) {

        if (compareAndSetState(0, acquires)) { // There is no judgment hasqueuedtoraise ()

            setExclusiveOwnerThread(current);

            return true; }}else if (current == getExclusiveOwnerThread()) {

        int nextc = c + acquires;

        if (nextc < 0// overflow

        throw new Error("Maximum lock count exceeded");

        setState(nextc);

        return true;

    }

    return false;

}

Copy the code

The only difference between the lock() method of a fair lock and that of an unfair lock is that the fair lock has a constraint on the acquisitionof the lock: hasqueuedToraise () is false. This method determines whether a thread is already queuing on the wait queue. This is the core difference between a fair lock and an unfair lock. If it is a fair lock, the current thread will not attempt to acquire the lock once a thread has queued. For an unfair lock, regardless of whether there is already a queue, the thread will try to acquire the lock, if not, queue again.

There is a special case to note here, for the tryLock() method, which does not adhere to the set fairness principle.

For example, when a thread executes the tryLock() method, once a thread releases the lock, the thread that is trylocked can acquire the lock, even if the lock mode is set to fair, and even if there are other threads waiting in the queue before it, simply saying that tryLock can cut the queue.

If you look at its source code, you will find:


public boolean tryLock(a) {

    return sync.nonfairTryAcquire(1);

}

Copy the code

NonfairTryAcquire () is called, indicating that the lock is unfair, regardless of whether the lock itself is fair.

Fair locks are acquired in the order that multiple threads apply for locks to achieve fair features. When unfair lock is added, it directly tries to obtain the lock without considering the queuing situation. Therefore, it is possible to obtain the lock first after application, but it also improves the overall efficiency.

Read-write lock

Before the read/write lock, we assume that we use a normal ReentrantLock. While we are thread-safe, we are wasting resources because if multiple reads are running at the same time, there is no thread-safe problem. We can allow multiple reads to run in parallel to improve program efficiency.

However, writes are not thread-safe, and can cause thread-safe problems if multiple threads write at the same time or read at the same time.

The overall idea is that it has two locks, the first lock is a write lock, after obtaining the write lock, can both read data and modify data, and the second lock is a read lock, after obtaining the read lock, can only view data, can not modify data. Read locks can be held by multiple threads, so multiple threads can view data at the same time.

Access rules for read and write locks

We follow the following acquisition rules when using read/write locks:

1. If one thread has occupied the read lock, the other thread can apply for the read lock successfully.

2. If a thread has occupied the read lock, if another thread applies for the write lock, the thread that applies for the write lock waits for the release of the read lock, because the read and write operations cannot be performed simultaneously.

3. If one thread has occupied the write lock, other threads must wait for the previous thread to release the write lock if they apply for the write lock.

To sum up: either one or more threads have a read lock, or one thread has a write lock, but not both. It can also be summarized as: read sharing, everything else is mutually exclusive (write mutually exclusive, read mutually exclusive, write mutually exclusive)

case

ReentrantReadWriteLock is an implementation class of ReadWriteLock. It has two main methods: readLock() and writeLock() to obtain read and write locks.


/** * description: demonstrates the use of read/write locks */

public class ReadWriteLockDemo {

private static final ReentrantReadWriteLock reentrantReadWriteLock = new ReentrantReadWriteLock(

false);

private static final ReentrantReadWriteLock.ReadLock readLock = reentrantReadWriteLock

.readLock();

private static final ReentrantReadWriteLock.WriteLock writeLock = reentrantReadWriteLock

.writeLock();

private static void read(a) {

readLock.lock();

try {

System.out.println(Thread.currentThread().getName() + "Got read lock, reading now");

Thread.sleep(500);

} catch (InterruptedException e) {

e.printStackTrace();

} finally {

System.out.println(Thread.currentThread().getName() + "Release read lock"); readLock.unlock(); }}private static void write(a) {

writeLock.lock();

try {

System.out.println(Thread.currentThread().getName() + "Got write lock, writing");

Thread.sleep(500);

} catch (InterruptedException e) {

e.printStackTrace();

} finally {

System.out.println(Thread.currentThread().getName() + "Release write lock"); writeLock.unlock(); }}public static void main(String[] args) throws InterruptedException {

new Thread(() -> read()).start();

new Thread(() -> read()).start();

new Thread(() -> write()).start();

newThread(() -> write()).start(); }}Copy the code

Conclusion ReentrantReadWriteLock

  • Jump the queue strategy
  1. Under the fair policy, as long as there are threads in the queue that are already queuing, queue jumping is not allowed.

2. Under unfair strategy:

1). If read lock is allowed to jump the queue, so as to read lock can be held at the same time by multiple threads, so may cause a steady stream of the thread has been cut in line behind the success, lead to read lock has been released in full, which can lead to write lock has been waiting for, in order to prevent the “hunger”, waiting for is to try to get the head of the queue nodes threads write locks, are not allowed to read lock cut in line

2). Write lock can jump the queue, because it is not easy to write locks to jump the queue is successful, write locks only in no other thread is currently holding read locks and write locks, can jump the queue, successful and write locks once cut in line failure will enter the waiting queue, so it is difficult to cause “hungry”, allows the write lock is jumped the queue in order to improve the efficiency.

  • Promotion and demotion strategy:

A write lock can only be downgraded to a read lock, but cannot be upgraded from a read lock to a write lock.

spinlocks

The spin lock.

“Spin” can be understood as “self-rotation,” where “spin” refers to a “loop,” such as a while loop or a for loop. “Spin” is the self repeating the cycle until the goal is achieved. Unlike normal locks, which block if the lock is not acquired

| – > contrast spin and spin the process of acquiring a lock

The benefits of | – > the spin lock

First, both blocking and waking up threads are expensive, and if synchronizing the contents of a code block is not complex, the overhead of switching threads may be greater than the actual business code execution.

In many situations, it may be our synchronized code block content is not much, so I need the execution time is very short, if we just for this time is to switch the thread state, then let the thread actually not switch state, but allow it to attempts to acquire a spin lock, waiting for the other thread lock is released, sometimes I just need to wait for a moment, Context switching and other overhead can be avoided and efficiency can be improved.

A spin lock is a loop that continually tries to acquire the lock, keeping the thread in a Runnable state and saving the overhead of thread state switching

Case in point: reentrant spin locks


package lesson27;

 

import java.util.concurrent.atomic.AtomicReference;

import java.util.concurrent.locks.Lock;

 

/** * Description: Implement a reentrant spin lock */

public class ReentrantSpinLock  {

 

    private AtomicReference<Thread> owner = new AtomicReference<>();

 

    // Reentrant times

    private int count = 0;

 

    public void lock(a) {

        Thread t = Thread.currentThread();

        if (t == owner.get()) {

            ++count;

            return;

        }

        // Spin the lock

        while(! owner.compareAndSet(null, t)) {

            System.out.println("Spinning."); }}public void unlock(a) {

        Thread t = Thread.currentThread();

        // Only the thread holding the lock can unlock it

        if (t == owner.get()) {

            if (count > 0) {

                --count;

            } else {

                // There is no CAS operation here because there is no race because only thread holders can unlock

                owner.set(null); }}}public static void main(String[] args) {

        ReentrantSpinLock spinLock = new ReentrantSpinLock();

        Runnable runnable = new Runnable() {

            @Override

            public void run(a) {

                System.out.println(Thread.currentThread().getName() + "Start trying to get a spin lock.");

                spinLock.lock();

                try {

                    System.out.println(Thread.currentThread().getName() + "Got the spin lock.");

                    Thread.sleep(4000);

                } catch (InterruptedException e) {

                    e.printStackTrace();

                } finally {

                    spinLock.unlock();

                    System.out.println(Thread.currentThread().getName() + "Release the spin lock."); }}}; Thread thread1 =new Thread(runnable);

        Thread thread2 = newThread(runnable); thread1.start(); thread2.start(); }}Copy the code
  • disadvantages

The biggest disadvantage is that while it avoids the overhead of thread switching, it also introduces new overhead as it constantly has to try to acquire the lock. If the lock can never be released, then this attempt is a futile attempt and a waste of processor resources. In other words, although the cost of a spinlock is lower than that of a thread switch at first, it increases over time and can even exceed the cost of a thread switch later on.

  • Applicable scenario

Spinlocks are suitable for scenarios where concurrency is not particularly high and where critical sections are short, so we can improve efficiency by avoiding thread switching

However, if the critical area is large and the thread takes a long time to release the lock, then spin-locking is not suitable because the spin will always use up the CPU but can’t get the lock, wasting resources

The JVM lock optimization

Compared to JDK 1.5, HotSopt virtual machine in JDK 1.6 has many optimizations for synchronized built-in lock performance, including adaptive spin, lock elimination, lock coarser, biased lock, lightweight lock, and so on. With these optimizations, the performance of synchronized locks has been greatly improved, and the specific optimizations are described below

Adaptive spin lock

Adaptive spin locks were introduced in JDK 1.6 to address the problem of long spins. Adaptive means that the timing of the spin is no longer fixed, but is determined by factors such as the success rate of recent spin attempts, the failure rate, and the status of the current lock owner. The duration of the spin changes, and the spin lock becomes “smart”. For example, if a recent attempt to spin a lock was successful, the next attempt might be to continue spinning and allow longer spins. But if you have recently failed to spin a lock, you might omit the spin process to reduce unwanted spin and improve efficiency

Lock elimination


@Override

public synchronized StringBuffer append(Object obj) {

    toStringCache = null;

    super.append(String.valueOf(obj));

    return this;

}

Copy the code

As you can see from the code, this method is a synchronized method modified by synchronized because it may be used by multiple threads simultaneously.

In most cases, however, it is only used in one thread, and if the compiler is certain that the StringBuffer object is only used in one thread, it is thread-safe, and our compiler optimizes to eliminate synchronized. The operation of locking and unlocking is eliminated in order to increase overall efficiency.

Lock coarsening

If we release the lock and then do nothing, we acquire it again, as shown in this code:


public void lockCoarsening(a) {

    synchronized (this) {

        //do something

    }

    synchronized (this) {

        //do something

    }

    synchronized (this) {

        //do something}}Copy the code

Actually this kind of release and retrieves the lock is completely unnecessary, if we expand the synchronization area, also is only in the most began to a lock, and directly in the final release, you can add the middle these meaningless unlock and lock the process of elimination, rather then combine several synchronized blocks into a larger synchronized block. The advantage of this is that locks are not frequently applied and released while the thread is executing the code, which reduces the performance overhead.

However, the side effect of doing this is that we make the synchronization area bigger. If we do the same in the loop, as the code shows:


for (int i = 0; i < 1000; i++) {

    synchronized (this) {

        //do something}}Copy the code

That is, at the beginning of the first loop, we start to expand the synchronization area and hold the lock until the end of the last loop, which releases the lock and causes other threads to be unable to acquire the lock for a long time. Therefore, lock coarsening here does not apply to cyclic scenarios, but only to non-cyclic scenarios.

Lock coarcing is turned on by default and can be turned off with -xx: -Eliminatelocks.

Bias lock/lightweight lock/heavyweight lock

These three types of locks refer specifically to the synchronized lock state, which is indicated by the mark word in the object header