1. IntroductionIn the previous thread series article, we introduced the synchronized and volatile keywords, which can solve the problem of thread synchronization, but they cannot solve the problem of coordination and communication between threads. To give a simple example, thread A is responsible for adding the value of the int variable i to 10000, and then notifies thread B to print out the result. How to achieve this? One of the simplest ways is that thread B continuously checks whether the condition is met through polling while(i == 10000). Although this method can meet the requirements, it also brings another problem: the while() operation in thread B will not release CPU resources, causing the CPU to always perform judgment operations on this method, greatly wasting CPU resources. We know that CPU resources are very expensive, because CPU resources are used not only by the current application, but also by many other applications. If the polling time is released and used by other threads, the running efficiency of the application can be significantly improved. For example, after thread A completes the operation, it notifies thread B to perform subsequent operations. Thread B does not need to complete the coordination between threads through polling. Isn't this better? In the parent class of Java, that is, the Object class, there are three methods: wait(), notify(), and notifyAll(), which can realize communication between threads. If you have never used multithreading, you may not use these methods. Let's take a look at how to use them! 2. Method Introduction
The wait() method, as the name suggests, means waiting. Its function is to put the thread executing the current code into a blocked state, put the current thread into the "pre-execution queue", and stop executing the code where wait() is located until it is notified or interrupted. However, there is a prerequisite. Before calling the wait() method, the thread must obtain the lock of the object. Therefore, the wait() method can only be called in the synchronized method/synchronized code block modified by synchronized. At the same time, after the wait() method is executed, the acquired object lock will be released immediately for other threads to use. The current thread is blocked and enters the waiting state. As for why wait() has a blocking effect, its internal mechanism is very complicated and is mainly implemented by the C code of the JVM. You just need to understand it.
The notify() method, as the name suggests, means notification. Its function is to reawaken the waiting threads under the same monitor. If there are multiple threads waiting, a waiting thread is randomly selected and a notify() is sent to it, making it wait to acquire the object lock of the object. Note "waiting to acquire the object lock of the object", which means that even if the notification is received, the waiting thread will not immediately acquire the object lock, and must wait for the thread of the notify() method to release the lock. The calling environment is the same as wait(). notify() must also be called in a synchronized method/synchronized code block modified by synchronized.
The notifyAll() method, as the name implies, also means notification. Its function is to reawaken all waiting threads under the same monitor. The notify() method will only wake up one thread randomly, while the notifyAll() method will wake up all threads at once. Generally speaking, the notifyAll() method is safer, because when our code logic is not well considered, using notify() will result in only waking up one thread, while other threads may wait forever and cannot wake up. The calling environment is the same as notify(). notifyAll() must also be called in a synchronized method/synchronized code block modified by synchronized. The three methods can be summarized as follows:
2.1、Introduction to wait/notify/notifyAllUsually the wait() method is often used in conjunction with notify() or notifyAll(). Let's look at a simple example. Running the service, the output is as follows: From the log, we can see that threadA starts first, then enters a blocked state. After 3 seconds, threadB is started. After the execution is completed, threadA is notified that it can acquire the object lock and the execution is completed. This is basically the coordination and communication between threads. If we increase the number of threadA threads to 5, let's take a look at the running results. Running the service, the output is as follows: From the log, we can clearly see that when multiple threads are in a waiting state, calling the notify() method will only wake up one of the waiting threads; at the same time, the service cannot be closed because the remaining 4 threads are always in a blocked state. Suppose we change the lock.notify() method in the MyThreadB class to the lock.notifyAll() method and see what the effect is. Running the service, the output is as follows: It can be clearly seen from the log that after 3 seconds all waiting threads are awakened and the service ends. 2.2. Introduction to wait lock releaseIn multi-threaded programming, you should always pay attention to locks because they play an important role in whether the current code is executed safely. As we mentioned above, calling the wait() method not only blocks the thread and puts it into a waiting state, but also releases the lock. We can see this by looking at a simple example. Running the service, the output is as follows: It can be clearly seen from the log results that after one of the two threads calls lock.wait(), it enters a blocked state and releases the object lock. The other thread obtains the lock and enters the synchronized code block, so wait begin is printed for both threads. There is also a sleep() method in the Thread class that can block the current thread, but there is a difference between them. The sleep() method will not allow the current thread to release the lock. Let's look at a simple example. Running the service, the output is as follows: From the log, it can be seen that the threads are not executed alternately, but serially. 2.3. Introduction to notify/notifyAll not releasing locksCorresponding to this are notify() and notifyAll(). Calling the notify() or notifyAll() method will not release the lock of the current thread. The lock will only be released when the synchronization method/synchronization code block is executed. Again, let's look at a simple example. Running the service, the output is as follows: It can be clearly seen from the log results that the two threads are not executed alternately, but serially. 2.4. IllegalMonitorStateExceptionAlthough the wait(), notify(), and notifyAll() methods are in the Object class and can be called directly by any class in theory, they cannot be called anywhere. If these three methods are called outside of a synchronized method or synchronized code block, a java.lang.IllegalMonitorStateException will be thrown when the program is running. Let's take a look at a simple example to understand. Run the program and throw an exception directly. Changing to notify() and notifyAll(), the running results are the same. 3. SummaryThis article mainly summarizes some knowledge about the coordination and communication technologies between threads. The wait(), notify(), and notifyAll() methods in the Object class can be used to achieve coordination and communication between threads, but they are only effective in synchronized methods/synchronized code blocks. If they are not called in synchronized methods/synchronized code blocks, a java.lang.IllegalMonitorStateException will be thrown. There are bound to be omissions in the article content, netizens are welcome to leave comments and point them out! IV. Reference1. Liao Xuefeng - Introduction to wait and notify 2. Cangjie in May - Introduction to wait() and notify()/notifyAll() |
<<: Why can't I ping the port number? How can I verify that the port number is available?
>>: The role of 5G in education: enabling distance learning and virtual labs
Alabama Children's Hospital is a large childr...
spinservers is a site under Majestic Hosting Solu...
The Chinese New Year has passed, but RAKsmart has...
A data center is a complex organization with many...
🌟 Opening: You think you know TCP? Actually... Co...
The three major operators are accelerating the sp...
AlphaVPS is a foreign hosting company established...
Today, Google announced that it has acquired Owlc...
In the process of implementing the integrated wir...
[[428404]] This article is reprinted from the WeC...
Preface Traditional IP packet switching networks ...
1. The past and present of SRT SRT is the acronym...
This month, Moack.co.kr launched a specially conf...
Speaking of the Internet of Vehicles, I believe e...