www.cnblogs.com/jiangyang/p…
Abstract: Java implements producer-consumer problem and reader-writer problem
1. Producer-consumer issues
Producer consumer problem is one of the classic problems that cannot be avoided when studying multithreaded programs. It describes that there is a buffer as a warehouse, producers can put products into the warehouse, and consumers can take products from the warehouse. The solutions to the producer/consumer problem can be divided into two categories :(1) adopt some mechanism to protect the synchronization between producer and consumer; (2) Build a pipeline between producers and consumers. The first method has higher efficiency, and easy to implement, the code can be controlled better, belongs to the common mode. The second kind of pipeline buffer is not easy to control, the transmitted data object is not easy to encapsulate, etc., so it is not practical.
The core of the synchronization problem is how to ensure the integrity of the same resource concurrently accessed by multiple threads. The common synchronization method is to use a signal or locking mechanism to ensure that the resource can be accessed by at most one thread at any time. Java language implements complete objectification in multithreaded programming and provides good support for synchronization mechanism. There are five methods that support synchronization in Java, the first four of which are synchronous methods and one is pipe methods.
- Wait ()/notify() methods
- Await ()/signal() methods
- BlockingQueue blocks the queue method
- Semaphore method
- PipedInputStream / PipedOutputStream
1.1 Wait ()/notify()
The wait()/nofity() methods are two methods of the base Object class, which means that all Java classes have these methods, so we can implement synchronization for any Object.
Wait () method: When the buffer is full/empty, the producer/consumer thread stops its execution, waives the lock, puts itself in a wait state, and lets other threads execute.
Notify () : When a producer/consumer puts/takes a product out of the buffer, it gives an executable notification to other waiting threads and waives the lock, leaving itself in a wait state.
We have four producers and four consumers
package test;
public class Hosee {
private static Integer count = 0;
private final Integer FULL = 10;
private static String LOCK = "LOCK";
class Producer implements Runnable {
@Override
public void run() {
for (int i = 0; i < 10; i++) {
try {
Thread.sleep(3000);
} catch (Exception e) {
e.printStackTrace();
}
synchronized (LOCK) {
while (count == FULL) {
try {
LOCK.wait();
} catch (Exception e) {
e.printStackTrace();
}
}
count++;
System.out.println(Thread.currentThread().getName() + "The producers produce, so there's a total of + count);
LOCK.notifyAll();
}
}
}
}
class Consumer implements Runnable {
@Override
public void run() {
for (int i = 0; i < 10; i++) {
try {
Thread.sleep(3000);
} catch (InterruptedException e1) {
e1.printStackTrace();
}
synchronized (LOCK) {
while (count == 0) {
try {
LOCK.wait();
} catch (Exception e) {
}
}
count--;
System.out.println(Thread.currentThread().getName() + "Consumers are spending, so there's a total of+ count); LOCK.notifyAll(); } } } } public static void main(String[] args) throws Exception { Hosee hosee = new Hosee(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); }}Copy the code
(Note that notify and wait are the same as LOCK. In this example, LOCK is used.)
Partial print results:
Since producer and consumer statements are consistent, the maximum is around 2, and when one consumer is reduced, it increases to 10.
1.2 await()/signal() methods
Await ()/signal() vs. wait()/notify()
- Wait () and notify() must be used in synchronized blocks because they can only be done when the Lock of the current object is acquired otherwise an exception is raised and await() and signal() are generally used with Lock().
- Wait is an Object method, and await is only part of the class, such as Condition.
- Await ()/signal() and the newly introduced locking mechanism, Lock, hook directly with greater flexibility.
So why synchronized?
1.2.1 Improvement of synchronized
Synchronized isn’t perfect, and it has some functional limitations — it can’t interrupt a thread that’s waiting for a lock, it can’t vote for a lock, and it can’t get a lock if you don’t want to wait. Synchronization also requires that the lock be released only in the same stack frame from which the lock was acquired, which is fine in most cases (and interacts well with exception handling), but there are cases where non-block locking is more appropriate.
1.2.2 already class
Java. Util. Concurrent. Lock the lock framework is an abstraction to lock, it allows the realization of the lock as a Java class, rather than as a language features to achieve (more object oriented). This leaves room for multiple implementations of Lock, which may have different scheduling algorithms, performance characteristics, or locking semantics. The ReentrantLock class implements Lock, which has the same concurrency and memory semantics as synchronized, but adds features like Lock voting, timed Lock waiting, and interruptible Lock waiting. In addition, it provides better performance in cases of intense contention. (In other words, when many threads want to access a shared resource, the JVM can spend less time scheduling threads and more time executing threads.)
What does a Reentrant lock mean? In simple terms, it has an acquisition counter associated with the lock, and if one of the threads that owns the lock gets it again, the acquisition counter is incremented by one, and then the lock needs to be released twice before it can actually be released (reentrant). This mimics the semantics of synchronized; If a thread enters a synchronized block protected by a monitor it already owns, the thread is allowed to continue. When a thread exits a second (or subsequent) synchronized block, the lock is not released, except when the thread exits the first synchronized block protected by the monitor it entered. Before releasing the lock.
A brief explanation of reentrant locking:
public class Child extends Father implements Runnable{ final static Child child = new Child(); Public static void main(String[] args) {for (int i = 0; i < 50; i++) {
new Thread(child).start();
}
}
public synchronized void doSomething() {
System.out.println("1child.doSomething()");
doAnotherThing(); // Call other synchronized methods in your class} private synchronized voiddoAnotherThing() { super.doSomething(); // Call the synchronized method system.out.println ("3child.doAnotherThing()");
}
@Override
public void run() {
child.doSomething();
}
}
class Father {
public synchronized void doSomething() {
System.out.println("2father.doSomething()"); }}Copy the code
When executing child.dosomething, the thread acquires the lock of the child object. When executing doAnotherThing in the doSomething method, the thread requests the lock of the Child object again because synchronized is a re-entrant lock. If the parent doSomething method is executed in doAnotherThing, the child object will be locked for a third time. If the parent doSomething method is executed in doAnotherThing, the child object will be locked for a third time.
When you look at the following code example, you can see one significant difference between Lock and synchronized — Lock must be released ina finally block. Otherwise, if the protected code throws an exception, the lock may never be released! This distinction may not seem like much, but in fact, it is extremely important. Forgetting to release the lock in the finally block can leave a time bomb in your program that one day, when it explodes, takes a lot of effort to find the source. With synchronization, the JVM ensures that the lock is automatically released.
Lock lock = new ReentrantLock();
lock.lock();
try {
// update object state
}
finally {
lock.unlock();
}Copy the code
In addition, the ReentrantLock implementation under contention is much more scalable than the current synchronized implementation. (Contention performance of Synchronized will most likely improve in future JVM releases.) This means that when many threads are competing for the same lock, the overall overhead of ReentrantLock is usually much less than that of synchronized.
1.2.3 When to replace synchronized with ReentrantLock
Synchronized is inefficient in Java1.5. Because this is a heavyweight operation that calls the operation interface, it is possible that locking will consume more system time than operations other than locking. By contrast, using Java provided Lock objects provides better performance. But with Java1.6, something changed. Synchronized is semantically clear and can be optimized for many things, including adaptive spin, lock elimination, lock coarser, lightweight lock, biased lock, and so on. Synchronized performance is not worse than Lock on Java1.6. Officials also say they’re more supportive of Synchronized, and there’s room for improvement in future releases.
So ReentrantLock is used when you really need some features that synchronized doesn’t have, such as time lock waits, interruptible lock waits, block-free structured locks, multiple condition variables, or lock voting. ReentrantLock also has scalability benefits and should be used in cases of high contention, but keep in mind that most synchronized blocks almost never have contention, so high contention can be put aside. I recommend developing with synchronized until it is proven that synchronized is not appropriate, rather than just assuming that “performance will be better” if you use ReentrantLock. Keep in mind that these are advanced tools for advanced users. (Also, true power users like to choose the simplest tool they can find until they decide that simple tools don’t work.) . As always, get things right first and then decide if it’s necessary to do them faster.
1.2.4 Next we use ReentrantLock to implement the producer-consumer problem
package test;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
public class Hosee {
private static Integer count = 0;
private final Integer FULL = 10;
final Lock lock = new ReentrantLock();
final Condition NotFull = lock.newCondition();
final Condition NotEmpty = lock.newCondition();
class Producer implements Runnable {
@Override
public void run() {
for (int i = 0; i < 10; i++) {
try {
Thread.sleep(3000);
} catch (Exception e) {
e.printStackTrace();
}
lock.lock();
try {
while (count == FULL) {
try {
NotFull.await();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
count++;
System.out.println(Thread.currentThread().getName()
+ "The producers produce, so there's a total of + count);
NotEmpty.signal();
} finally {
lock.unlock();
}
}
}
}
class Consumer implements Runnable {
@Override
public void run() {
for (int i = 0; i < 10; i++) {
try {
Thread.sleep(3000);
} catch (InterruptedException e1) {
e1.printStackTrace();
}
lock.lock();
try {
while (count == 0) {
try {
NotEmpty.await();
} catch (Exception e) {
// TODO: handle exception
e.printStackTrace();
}
}
count--;
System.out.println(Thread.currentThread().getName()
+ "Consumers are spending, so there's a total of+ count); NotFull.signal(); } finally { lock.unlock(); } } } } public static void main(String[] args) throws Exception { Hosee hosee = new Hosee(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); }}Copy the code
The result is similar to the first one. The above code uses two conditions, but it is possible to use one, just signalAll ().
1.3 BlockingQueue blocks the queue method
BlockingQueue, a new addition to JDK5.0, is a queue that is already synchronized internally with our second await()/signal() method. It can specify the size of the object when it is generated. It uses the put() and take() methods for blocking operations.
The put() method: Similar to our producer thread above, blocks automatically when it reaches its maximum capacity.
Take () method: Similar to our consumer thread above, blocks automatically at zero capacity.
package test;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
public class Hosee {
private static Integer count = 0;
final BlockingQueue<Integer> bq = new ArrayBlockingQueue<Integer>(10);
class Producer implements Runnable {
@Override
public void run() {
for (int i = 0; i < 10; i++) {
try {
Thread.sleep(3000);
} catch (Exception e) {
e.printStackTrace();
}
try {
bq.put(1);
count++;
System.out.println(Thread.currentThread().getName()
+ "The producers produce, so there's a total of + count);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
class Consumer implements Runnable {
@Override
public void run() {
for (int i = 0; i < 10; i++) {
try {
Thread.sleep(3000);
} catch (InterruptedException e1) {
e1.printStackTrace();
}
try {
bq.take();
count--;
System.out.println(Thread.currentThread().getName()
+ "Consumers are spending, so there's a total of+ count); } catch (Exception e) { // TODO: handle exception e.printStackTrace(); } } } } public static void main(String[] args) throws Exception { Hosee hosee = new Hosee(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); }}Copy the code
This BlockingQueue is difficult to demonstrate in code because the put() and take() methods cannot be synchronized with the output statement. You can implement BlockingQueue yourself (BlockingQueue is implemented with await()/signal()). So you’ll find a mismatch in the output.
Such as: When the buffer is full, the producer calls the await() method inside the put() operation, aborting the execution of the thread. Then the consumer thread executes, calling the take() method, and inside the take() method calls the signal() method, notifying the producer that the thread can execute, The producer println() is executed before the consumer println() is run, so there is an output mismatch.
With BlockingQueue you can rest assured that there is nothing wrong with it, just synchronization between it and other objects.
1.4 Semaphore method
Semaphore is a token that allows implementation to be set. Maybe there’s one, maybe there’s 10 or more.
Whoever acquires a token can execute it; if there is no token, wait.
The token must be released after execution, otherwise it will run out quickly and no other thread will be able to get the token and execute.
package test;
import java.util.concurrent.Semaphore;
public class Hosee
{
int count = 0;
final Semaphore notFull = new Semaphore(10);
final Semaphore notEmpty = new Semaphore(0);
final Semaphore mutex = new Semaphore(1);
class Producer implements Runnable
{
@Override
public void run()
{
for(int i = 0; i < 10; i++) { try { Thread.sleep(3000); } catch (Exception e) { e.printStackTrace(); } try { notFull.acquire(); // The order cannot be reversed, otherwise it will cause a deadlock. mutex.acquire(); count++; System.out.println(Thread.currentThread().getName() +"The producers produce, so there's a total of + count);
}
catch (Exception e)
{
e.printStackTrace();
}
finally
{
mutex.release();
notEmpty.release();
}
}
}
}
class Consumer implements Runnable
{
@Override
public void run()
{
for(int i = 0; i < 10; i++) { try { Thread.sleep(3000); } catch (InterruptedException e1) { e1.printStackTrace(); } try { notEmpty.acquire(); // The order cannot be reversed, otherwise it will cause a deadlock. mutex.acquire(); count--; System.out.println(Thread.currentThread().getName() +"Consumers are spending, so there's a total of+ count); } catch (Exception e) { e.printStackTrace(); } finally { mutex.release(); notFull.release(); } } } } public static void main(String[] args) throws Exception { Hosee hosee = new Hosee(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); }}Copy the code
Note that notFull.acquire() and mutex.acquire() are not interchangeable, and if a mutex is acquired before waiting, a deadlock will occur.
1.5 PipedInputStream/PipedOutputStream
This class, located in the java.io package, is the simplest solution to the synchronization problem, with one thread writing data to the pipe and another reading data from the pipe, thus creating a producer/consumer buffer programming pattern. PipedInputStream/PipedOutputStream can only be used for multithreaded mode for single thread may lead to a deadlock.
package test;
import java.io.IOException;
import java.io.PipedInputStream;
import java.io.PipedOutputStream;
public class Hosee {
final PipedInputStream pis = new PipedInputStream();
final PipedOutputStream pos = new PipedOutputStream();
{
try {
pis.connect(pos);
} catch (IOException e) {
e.printStackTrace();
}
}
class Producer implements Runnable {
@Override
public void run() {
try{
while(true){
int b = (int) (Math.random() * 255);
System.out.println("Producer: a byte, the value is " + b);
pos.write(b);
pos.flush();
}
}catch(Exception e){
e.printStackTrace();
}finally{
try{
pos.close();
pis.close();
}catch(IOException e){
System.out.println(e);
}
}
}
}
class Consumer implements Runnable {
@Override
public void run() {
try{
while(true){
int b = pis.read();
System.out.println("Consumer: a byte, the value is "+ String.valueOf(b)); } }catch(Exception e){ e.printStackTrace(); }finally{ try{ pos.close(); pis.close(); }catch(IOException e){ System.out.println(e); } } } } public static void main(String[] args) throws Exception { Hosee hosee = new Hosee(); new Thread(hosee.new Producer()).start(); new Thread(hosee.new Consumer()).start(); }}Copy the code
As with blocking queues, since the read()/write() methods are not necessarily synchronized with the output methods, there is a mismatch in the output results, and to make the results more obvious, there is only one consumer and one producer.
2. Reader writer questions
The reader-writers problem is also a classical concurrent programming problem, which is a synchronization problem that often occurs. Data (files, records) in a computer system is often shared by multiple processes, but some of these processes may only require the data to be read (called readers). Other processes require modification of data (called Writer). In terms of sharing data, Reader and Writer are two groups of concurrent processes that share a set of data areas.
(1) Allow multiple readers to read at the same time;
(2) Readers and writers are not allowed to operate simultaneously;
(3) Multiple writers are not allowed to operate simultaneously.
The synchronization problem of Reader and Writer can be divided into three types: Reader first, weak Writer first (fair play), and strong Writer first, which are handled differently.
Let’s first look at some of the ways that Java can implement the reader writer problem in the context of fair play
2.1 read-write lock
ReentrantReadWriteLock uses two types of locks: one for a read lock and one for a write lock. A write lock can be accessed only when there is no write lock from another thread, no write request is made, or there is a write request, but the writing thread and the holding thread are the same thread. No read locks from other threads No write locks from other threads
When you go to ReentrantReadWriteLock, the first thing to do is to steer clear of ReentrantLock. It and the latter are separate implementations with no inheritance or implementation relationship to each other. Then we summarize the features of the locking mechanism:
- The WriteLock inside the reentrant (described above in ReentrantLock) aspect can get ReadLock, but ReadLock’s attempt to get WriteLock in return never does.
- WriteLock can be demoted to ReadLock by acquiring WriteLock, acquiring ReadLock, and releasing WriteLock. The thread holds the ReadLock. Conversely, ReadLock cannot upgrade to WriteLock. Why? See (1), hehe.
- ReadLock can be held by multiple threads and in effect excludes any WriteLock, which is completely mutually exclusive. This feature is most important because for data structures with high read frequencies and relatively low writes, such lock synchronization can increase concurrency.
- Both ReadLock and WriteLock support interrupts and have the same semantics as ReentrantLock.
- WriteLock support Condition and consistent with already semantics, whereas ReadLock cannot use Condition, otherwise throw UnsupportedOperationException anomalies.
Take a look at the two constructors of the ReentrantReadWriteLock class
public ReentrantReadWriteLock() {
this(false);
}
/**
* Creates a new {@code ReentrantReadWriteLock} with
* the given fairness policy.
*
* @param fair {@code true} if this lock should use a fair ordering policy
*/
public ReentrantReadWriteLock(boolean fair) {
sync = (fair)? new FairSync() : new NonfairSync();
readerLock = new ReadLock(this);
writerLock = new WriteLock(this);
}Copy the code
The fair parameter indicates whether a fair or unfair read-write lock is created. Preemptive or non-preemptive.
Fair and unfair: Fair means that locks are allocated in accordance with the sequence in which locks are acquired by the thread. Locks are allocated in accordance with the sequence in FIFO. Unfair means that the order in which locks are acquired is unnecessary, and that the later thread may acquire the lock first, which may result in some threads never acquiring the lock.
Why does fair lock affect performance? From code, we can see that fair lock is just an extra check at the head of the queue. If not, where does it affect performance? If it is an intruders thread, it will queue at the back and wait for the previous node to wake up (parking), which is bound to add a lot of paking and unparking operations than unfair lock
The general application scenario is: if there are multiple reader threads and one writer thread, and the writer thread needs to block the reader thread during operation, then the fair lock should be used. Otherwise, the writer thread may not get the lock, causing the thread to starve.
A quick word about lock degradation
Reentrant also allows you to degrade from a write lock to a read lock by acquiring the write lock, then acquiring the read lock, and finally releasing the write lock. However, upgrading from a read lock to a write lock is not possible.
rwl.readLock().lock();
if(! cacheValid) { // Must releaseread lock before acquiring write lock
rwl.readLock().unlock();
rwl.writeLock().lock();
if(! cacheValid) { data = ... cacheValid =true; } rwl.readLock().lock(); rwl.writeLock().unlock(); // downgrade: acquire read lock and release write lock}Copy the code
Let’s implement the reader writer problem with read/write locks
import java.util.Random;
import java.util.concurrent.locks.ReentrantReadWriteLock;
public class ReadWriteLockTest {
public static void main(String[] args) {
final Queue3 q3 = new Queue3();
for (int i = 0; i < 3; i++) {
new Thread() {
public void run() {
while (true) {
q3.get();
}
}
}.start();
}
for (int i = 0; i < 3; i++) {
new Thread() {
public void run() {
while (true) { q3.put(new Random().nextInt(10000)); } } }.start(); } } } class Queue3 { private Object data = null; // Share data. Only one thread can write the data, but multiple threads can read the data simultaneously. private ReentrantReadWriteLock rwl = new ReentrantReadWriteLock(); public voidget() { rwl.readLock().lock(); System.out.println(thread.currentThread ().getName() +)" be ready to read data!");
try {
Thread.sleep((long) (Math.random() * 1000));
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()
+ "have read data :"+ data); rwl.readLock().unlock(); } public void put(Object data) {rwl.writelock ().lock(); System.out.println(thread.currentThread ().getName() +)" be ready to write data!");
try {
Thread.sleep((long) (Math.random() * 1000));
} catch (InterruptedException e) {
e.printStackTrace();
}
this.data = data;
System.out.println(Thread.currentThread().getName()
+ " have write data: "+ data); rwl.writeLock().unlock(); // Release write lock}}Copy the code
Running results:
Thread-0 be ready to read data!
Thread-1 be ready to read data!
Thread-2 be ready to read data!
Thread-0have read data :null
Thread-2have read data :null
Thread-1have read data :null
Thread-5 be ready to write data!
Thread-5 have write data: 6934
Thread-5 be ready to write data!
Thread-5 have write data: 8987
Thread-5 be ready to write data!
Thread-5 have write data: 8496Copy the code
2.2 Semaphore Semaphore
In 1.4, we introduced the use of semaphores to implement the producer-consumer problem. Now we will use semaphores to implement the reader-writer problem
package test;
import java.util.Random;
import java.util.concurrent.Semaphore;
public class ReadWrite
{
public static void main(String[] args)
{
final Queue3 q3 = new Queue3();
for (int i = 0; i < 3; i++)
{
new Thread()
{
public void run()
{
while (true)
{
try
{
Thread.sleep((long) (Math.random() * 1000));
}
catch (InterruptedException e)
{
e.printStackTrace();
}
q3.get();
}
}
}.start();
}
for (int i = 0; i < 3; i++)
{
new Thread()
{
public void run()
{
while (true) { try { Thread.sleep((long) (Math.random() * 1000)); } catch (InterruptedException e) { e.printStackTrace(); } q3.put(new Random().nextInt(10000)); } } }.start(); } } } class Queue3 { private Object data = null; // Share data. Only one thread can write the data, but multiple threads can read the data simultaneously. private Semaphore wmutex = new Semaphore(1); private Semaphore rmutex = new Semaphore(2); private int count = 0; public voidget()
{
try
{
rmutex.acquire();
if(count == 0) wmutex.acquire(); // prevent the writer from writing count++ when the first reader attempts to read the database; System.out.println(Thread.currentThread().getName() +" be ready to read data!");
try
{
Thread.sleep((long) (Math.random() * 1000));
}
catch (InterruptedException e)
{
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()
+ "have read data :" + data);
count--;
if (count == 0)
wmutex.release();
rmutex.release();
}
catch (Exception e)
{
e.printStackTrace();
}
}
public void put(Object data)
{
try
{
wmutex.acquire();
System.out.println(Thread.currentThread().getName()
+ " be ready to write data!");
try
{
Thread.sleep((long) (Math.random() * 1000));
}
catch (InterruptedException e)
{
e.printStackTrace();
}
this.data = data;
System.out.println(Thread.currentThread().getName()
+ " have write data: "+ data); } catch (Exception e) { e.printStackTrace(); } finally { wmutex.release(); }}}Copy the code
Semaphores alone cannot solve the reader/writer problem. The count counter (CountDownLatch can be used instead) must be introduced to count the read process. Count is used in conjunction with wMUtex to enable simultaneous reading and writing. If count is 0, the read process starts, and the write process blocks (wmutex is acquired by the read process). If count is not 0, there are multiple read processes, and wMUtex is not needed because the first read process has already acquired wMUtex. If count==0, the read process is complete. If count==0, the read process is complete. At this point, wMUtex is released and the writing process has a chance to acquire wMUtex. To prevent the reader from holding wMUtex all the time, it is better to let the reader sleep so that the writer has a chance to acquire wMUtex.
Conclusion:
The producer-consumer problem (5 kinds) and the reader-writer problem (2 kinds) with Java have been described, welcome to discuss and give different solutions. If anything goes wrong, please leave a message or text me. If I have any new ideas, I will timely supplement them.