After Jdk1.5, under the java.util.concurrent.locks package, there is a set of interfaces and classes that implement thread synchronization. When it comes to thread synchronization, everyone may think of the synchronized keyword,
This It is a built-in keyword in Java, used to handle thread synchronization. However, this keyword has many flaws and is not very convenient and intuitive to use, so Lock appears. Below, we
will compare Explain Lock.
Usually we encounter the following problems when using the synchronized keyword:
(1) Uncontrollability, unable to lock and release locks at will.
(2) The efficiency is relatively low. For example, we are currently reading two files concurrently. The reading and reading have no influence on each other. However, if synchronized is used for the read object to achieve synchronization,
So as long as one thread enters, other threads will have to wait.
(3) There is no way to know whether the thread has acquired the lock.
Lock can solve the above synchronized problems very well, and after jdk1.5, various locks are also provided, such as read-write locks, but there is one thing to note, using synchronized
When it is critical, there is no need to manually release the lock, but when using Lock, you must manually release the lock. Let's learn about Lock locks.
Lock is an upper-layer interface. Its prototype is as follows, providing a total of 6 methods:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
So how to use these methods? As we mentioned earlier, using Lock requires manual release of the lock. However, if an exception is thrown in the program, the lock cannot be released and may cause deadlock.
So when we use Lock , there is a fixed format, as follows:
1 2 3 4 5 6 7 |
|
Let’s look at a simple example, the code is as follows:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 |
|
1 , Function
lock and synchronized are both tools used in Java to solve thread safety issues.
2. Source
sychronized is a keyword in Java.
lock is an interface provided in the JUC package. This interface has many implementation classes, including our most commonly used ReentrantLock (reentrant lock).
3. Lock strength
sychronized can control the lock strength in two ways:
Modify the sychronized keyword at the method level.
is decorated on the code block.
Differences in lock objects:
If the lock object is a static object or a class object, then this lock is a global lock.
The lock object is a normal instance object, and the scope of this lock depends on the life cycle of this instance.
The strength of the lock is determined by the two methods lock() and unlock(). The code between the two methods is guaranteed to be thread-safe. The scope of the lock depends on the life cycle of the lock instance.
4. Flexibility
Lock is more flexible than sychronized.
lock can decide independently when to lock and release the lock. Just call the lock() and unlock() methods of lock.
sychronized Because it is a keyword, it cannot implement the non-blocking competition lock method. After one thread acquires the lock, other locks can only wait for that thread to release before they have a chance to acquire the lock.
5. Fair lock and unfair lock
Fair lock: Multiple threads obtain locks in the order in which they apply for locks. The threads will directly enter the queue to queue up, forever. Only the first person in the queue can get the lock.
Advantages: All threads can obtain resources and will not starve to death.
Disadvantages: Low throughput, except for the first thread in the queue, other threads will be blocked, and the CPU overhead of waking up blocked threads is high.
Unfair lock: When multiple threads acquire the lock, they will directly try to acquire it. If they cannot acquire it, they will enter the waiting queue. If they can acquire it, they will acquire the lock directly.
Advantages: It can reduce the overhead of CPU waking up threads, the overall throughput efficiency will be higher, and the CPU does not have to wake up all threads, which will reduce the number of awakened threads.
Disadvantages: The thread in the middle of the queue may not be able to obtain the lock or cannot obtain the lock for a long time, and eventually starve to death.
lock provides two mechanisms: fair lock and unfair lock (default unfair lock).
sychronized is an unfair lock.
6. Whether to release the lock due to exception
The release of the synchronized lock is passive and will only be released when the execution of the sychronized synchronized code block ends or an exception occurs.
When an exception occurs in the lock lock, the occupied lock will not be actively released. It must be released manually with unlock(), so we usually put the synchronization code block into try-catch and write unlock in finally. () method to avoid deadlock.
7. Determine whether the lock can be obtained
synchronized cannot.
lock provides a non-blocking competition lock method trylock(), and the return value is of type Boolean. It indicates that it is used to try to acquire the lock: it returns true if the acquisition is successful; it returns false if the acquisition fails. This method will return immediately no matter what.
8. Scheduling method
synchronized uses the wait, notify, and notifyAll methods of the object object itself, while lock uses Condition for scheduling between threads.
9. Whether it can be interrupted
synchronized can only wait for the lock to be released and cannot respond to interrupts.
You can use interrupt() to interrupt while waiting for the lock.
10. Performance
If the competition is not fierce, the performance is about the same; when the competition is fierce, the performance of lock will be better.
Lock lock can also use readwritelock to separate reading and writing, improving the efficiency of multi-threaded reading operations.
11. Synchronized lock upgrade
The synchronized code block is implemented by a pair of monitorenter/monitorexit instructions. The implementation of Monitor completely relies on the mutex lock inside the operating system. Because it requires switching from user mode to kernel mode, synchronization operation is an undifferentiated heavyweight operation.
So now the JVM provides three different locks: biased locks, lightweight locks, and heavyweight locks.
Biased lock:
When no competition occurs, biased lock is used by default. The thread will use the CAS operation to set the thread ID on the object header to indicate that the object is biased towards the current thread.
Purpose: In many application scenarios, the life cycle of most objects will be locked by at most one thread. Using biased locks can reduce the overhead when there is no competition.
Lightweight lock:
The JVM compares the threadID of the current thread and the threadID in the Java object header to see if it is consistent. If it is inconsistent (for example, thread 2 wants to compete for the lock object), then you need to check the record in the Java object header. Whether thread 1 is alive (the biased lock will not be actively released, so it is still the stored threadID of thread 1). If it is not alive, then the lock object is still a biased lock (the threadID in the object header is thread 2's); if it survives, then revoke Bias lock, upgraded to lightweight lock.
When other threads want to access resources with lightweight locks, spin lock optimization will be used to access the resources.
Purpose: There are not many threads competing for the lock object, and the threads do not hold the lock for a long time. Because blocking the thread requires the CPU to transfer from user mode to kernel mode, which is expensive. If the lock is released shortly after blocking, the gain is not worth the loss. Therefore, it is better not to block the thread at this time and let it spin to wait for the lock to be released.
Heavyweight lock:
If the spin fails, there is a high probability that the self-selection will fail again, so it is directly upgraded to a heavyweight lock to block threads and reduce CPU consumption.
When the lock is upgraded to a heavyweight lock, threads that have not grabbed the lock will be blocked and enter the blocking queue.
The above is the detailed content of How to use Lock in Java multithreading. For more information, please follow other related articles on the PHP Chinese website!