1.1 Relationship between number of CPU cores and number of threads 1.2 Time slice rotation mechanism (RR scheduling)1.3 Process and thread 1.3.1 What is a process? 1.3.2. What is a thread? 1.4, Concurrency and parallelism 1.4.1, what is concurrency 1.4.2, what is parallelism 1.5, synchronous and asynchronous 1.5.1, what is synchronous 1.5.2, what is asynchronous 2, multithreading 2.1, create multithreading 2.1.1, implement Runnable interface 2.1.2, inherit Thread class 2.1.3, implement Calla Ble interface 2.2, Thread termination 2.2.1, natural termination 2.2.2, Manual termination 3, Sharing and collaboration between threads 3.1, Sharing between threads 3.2, collaboration between threads 3.2.1, Nitify (), notifyAll(), wait() wait/notification mechanism
1. Basic Concepts
1.1. Relation between number of CPU cores and number of threads
Multi-core: Single-chip Multiprocessors (CMP). The idea is to integrate SMPS (symmetric Multiprocessors) from massively parallel processors into the same Chip, with each processor executing different processes in parallel. This reliance on multiple cpus to run programs simultaneously in parallel is an important direction for ultra-fast computing, called parallel processing,
Multi-threading: Multiple threads on the same processor execute synchronously and share the execution resources of the processor, which can maximize the realization of wide emission and out-of-order superscalar processing, improve the utilization of the processor’s computing components, and ease the memory access delay caused by data correlation or Cache failure.
The relationship between the two: the current CPU is basically multi-core, rarely see single-core CPU. The reason for increasing the number of cores is to increase the number of threads, because the operating system uses threads to perform tasks. Normally, there is a 1:1 relationship, meaning that a quad-core CPU has four threads. However, Intel introduced the hyper-threading technology, so that the number of cores and the number of threads form a 1:2 relationship.
1.2 Time Slice rotation mechanism (RR scheduling)
Definition: The system queues all ready processes on a first-in, first-out basis. The new process is added to the end of the ready queue. Whenever a process schedule is executed, the process scheduler always selects the queue leader of the ready queue to run a slice of time on the CPU. A time slice is a small unit of time, usually on the order of 10 to 100ms. When a process has used up its allotted time slice, the system timer issues a clock interrupt, and the scheduler stops the process and puts it at the end of the ready queue. The CPU is then assigned to the queue leader, which also runs a time slice, and so on.
According to the number of CPU core and the relationship between the number of threads a 1:1 relationship, if our mobile phones is a dual-core, so in principle we can only two threads, but in the actual development process is not the case, we may have opened a dozen threads “and” in the implementation, this is because the operating system provides the CPU time slice rotation this mechanism, It assigns each process a period of time, or slice of time, to execute alternately over a period of time.
Context switching time: Due to time slice rotation, there will be continuous switching between processes. Switching between processes involves saving and loading register values and memory images, updating tables and queues, and this process consumes time.
Time slice time setting: If the time slice is set too short, it will lead to too many processes constantly switching, because the switching process will produce text switching time, so the CPU efficiency will be reduced. If the time slice is set too long, it will also lead to poor response of relatively short interaction requests. Generally, it is reasonable to set the time slice at about 100ms.
1.3. Processes and Threads
1.3.1. What is a process?
A process is the smallest unit of running resources allocated by a program
A process is an independent unit of the operating system for resource allocation and scheduling. Resources include CPU, memory space, disk IO and so on. All threads of the same process share all the resources of the process, and processes are independent of each other.
1.3.2. What is a thread?
Threads are the smallest unit of CPU scheduling and must be process-dependent.
Thread is the entity of the process, is the basic unit of CPU scheduling and dispatching. Thread does not have system resources basically, but it has program counters, a group of registers, stacks and other indispensable resources in operation. The thread in the same process shares all resources owned by the process.
1.4. Concurrency and parallelism
1.4.1. What is concurrency
Concurrency is when several programs are running on the same CPU at any one time, but only one program is running on the processor at any one time.
Multiple threads per CPU
It’s tied to time, per unit time.
1.4.2 What is parallelism
Parallelism refers to the fact that several programs are running on several cpus at any one time. At any one time, multiple programs are running at the same time without interfering with each other.
Multiple threads multiple cpus
1.5. Synchronous and asynchronous
1.5.1 What is Synchronization
Synchronization: When a synchronous call is made, the call does not return until the result is returned.
It’s like I call my friend, and if you don’t pick up the phone, I’ll keep calling you, doing nothing but calling you until you pick up the phone.
1.5.2 what is asynchrony
Asynchronous: After an asynchronous call is made, the caller does not get the result immediately and the call returns.
I’ll send you a message to tell you I need you, and then I’ll go do my own thing. Call me back when you get the message. Of course, you don’t have to call me back.
Second, multi-threaded use
2.1. Create multiple threads
2.1.1. Implement Runnable interface
public static class newRunnable implements Runnable {
@Override
public void run(a) {
System.out.println("Runnable");
}
}
Copy the code
Call:
new Thread(new newRunnable()).start();
Copy the code
2.1.2 Inherit the Thread class
public static class newThread extends Thread {
@Override
public void run(a) {
super.run();
System.out.println("newThread");
}
}
Copy the code
Call:
new newThread().start();
Copy the code
1. Thread is an abstract concept for threads in Java. When we pass a new Thread, we actually create a Thread instance, and the operating system does not hook the Thread.
2. Start () causes a thread to enter the ready queue for CPU allocation, and then calls run().
3, the start () method cannot be repeated calls, otherwise you will be thrown IllegalThreadStateException anomalies.
2.1.3. Implement Callable interface
The Callable interface was provided in Java1.5 and can provide a return value after a task is completed.
public interface Callable<V> {
/ * *
* Computes a result, or throws an exception if unable to do so.
*
* @return computed result
* @throws Exception if unable to compute a result
* /
V call(a) throws Exception;
}
Copy the code
Callable and Runnable differ in that their call methods provide return values and throw exceptions.
Use:
public static class newCallable implements Callable<String> {
@Override
public String call(a) throws Exception {
System.out.println("newCallable");
Thread.sleep(3000);
return "Available after java1.5 to return results at the end of a task";
}
}
Copy the code
The call to Callable requires the FutureTask class, which is also available after Java1.5
FutureTask<String> futureTask = new FutureTask<>(new newCallable());
futureTask.run();
String result = null;
try {
result = futureTask.get();
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
Copy the code
Execution Result:
As you can see, we use FutureTask’s GET method, which blocks until the task is finished and does not return the value.
Take a look at FutureTask, which ultimately inherits from the Future interface. In addition to the get method described above, FutureTask provides the following methods to determine whether a task is complete and to cancel the task.
- 1.
boolean cancel(boolean mayInterruptIfRunning)
An attempt was made to cancel execution of the current task. If the task has been canceled, completed, or cannot be canceled for some other reason, the attempt will fail. Returns false.
If cancel(true) is called before the task is started, the task will never be executed.
If the task has been started, the parameter values determine whether the task should interrupt the thread executing the task.
True: Attempts to interrupt the task, returning true. False: the task is not interrupted and the execution continues, returning True.
-
Boolean isCancelled() Returns true if the task has been cancelled before its normal end
-
Boolean isDone() returns true if the task completes normally, is abnormal, or is canceled
-
4. V get() waits for the task to end and then gets the result. InterruptedException will be thrown if the task is interrupted while waiting and CancellationException will be thrown if the task is canceled. An ExecutionException will be thrown if an exception occurs during execution of the task.
-
5, V GET (long timeout, TimeUnit unit) task at most within the given time and return a result, if the task is not completed within the given time will throw a TimeoutException.
2.2. Terminate the thread
2.2.1 Natural termination
The thread terminates when the task is completed.
2.2.2 Manual termination
Suspend, resume, and stop operations correspond to the Thread APIS suspend(), resume(), and stop(). However, these apis are out of date, that is, not recommended, mainly because method calls do not guarantee the normal release of thread resources, which can cause other side effects.
Suspend () : After a call, the thread does not release resources it has already held (such as locks). Instead, it holds resources and goes to sleep, which can cause deadlock problems
Stop () : Terminating a thread does not guarantee that the thread’s resources will be released normally, usually without giving the thread a chance to complete the resource release, thus causing the program to work in an indeterminate state
A really safe way to terminate a thread is to use the interrupt() method
Since threads work cooperatively, when an interrupt() is used to terminate a thread, the thread does not terminate immediately. It only receives a termination notification and does so by checking that its interrupt flag is set to True. Of course, the thread can ignore it.
Judgment of interrupt flag bit:
1, the isInterrupted ()
This method checks whether the thread is interrupted, returning true if it has been interrupted, false otherwise, without changing the interrupt flag bit.
/ * *
* Tests whether this thread has been interrupted. The <i>interrupted
* status</i> of the thread is unaffected by this method.
*
* <p>A thread interruption ignored because a thread was not alive
* at the time of the interrupt will be reflected by this method
* returning false.
*
* @return <code>true</code> if this thread has been interrupted;
* <code>false</code> otherwise.
* @see #interrupted()
* @revised 6.0
* /
public boolean isInterrupted(a) {
return isInterrupted(false);
}
Copy the code
IsInterrupted example:
private static class UseThread extends Thread{
public UseThread(String name) {
super(name);
}
@Override
public void run(a) {
String threadName = Thread.currentThread().getName();
System.out.println(threadName+" interrupt start flag ="+isInterrupted());
while(! isInterrupted()){
System.out.println(threadName+" is running");
System.out.println(threadName+" inner interrupt flag ="+isInterrupted());
}
System.out.println(threadName+" interrupt end flag ="+isInterrupted());
}
}
Copy the code
Run the program above:
Start the thread, sleep for a microsecond and interrupt with interrupt()
public static void main(String[] args) throws InterruptedException {
Thread endThread = new UseThread("test isInterrupted");
endThread.start();
Thread.sleep(1);
endThread.interrupt();
}
Copy the code
Results:
As you can see UseThread isInterrupted() is always false. After the main thread executes endThread.interrupt(), the interrupt flag is set to true, breaking out of the loop and ending the run method. We later call isInterrupted() to print that the value of the interrupt flag remains true.
2, interrupted ()
Returns true if the thread has been interrupted. When the status is returned, this method clears the interrupt flag and sets it to false, so it returns false when called again (unless interrupted () is called again).
/ * *
* Tests whether the current thread has been interrupted. The
* <i>interrupted status</i> of the thread is cleared by this method. In
* other words, if this method were to be called twice in succession, the
* second call would return false (unless the current thread were
* interrupted again, after the first call had cleared its interrupted
* status and before the second call had examined it).
*
* <p>A thread interruption ignored because a thread was not alive
* at the time of the interrupt will be reflected by this method
* returning false.
*
* @return <code>true</code> if the current thread has been interrupted;
* <code>false</code> otherwise.
* @see #isInterrupted()
* @revised 6.0
* /
public static boolean interrupted(a) {
return currentThread().isInterrupted(true);
}
Copy the code
Interrupted () example:
We changed the code briefly:
private static class UseThread extends Thread{
public UseThread(String name) {
super(name);
}
@Override
public void run(a) {
String threadName = Thread.currentThread().getName();
while(! Thread.interrupted()){
System.out.println(threadName+" is running");
}
System.out.println(threadName+" interrupted end flag ="+Thread.interrupted());
}
}
Copy the code
As you can see, the run method loops until the Thread is interrupted, after which we call again and print thread.interrupted ()
Call:
public static void main(String[] args) throws InterruptedException {
Thread endThread = new UseThread("test interrupted");
endThread.start();
Thread.sleep(1);
endThread.interrupt();
}
Copy the code
The interrupt operation is also carried out after one microsecond sleep.
Results:
For example, if thread.interrupted () is true, the Thread ends the loop if thread.interrupted () is set to false. After thread.interrupted () is executed, the interrupt flag is cleared and set to false.
Note: deadlocked threads cannot be interrupted
A thread interrupts in a blocked state
Sleep, Thread. join, and Thread. wait are blocked, and the interrupt flag is true when the thread checks the interrupt flag. InterruptedException is thrown at these blocking method calls, and the thread’s interrupt flag is cleared immediately after the exception is thrown, reset to False.
We made this change in the previous example in the isInterrupted demo
private static class UseThread extends Thread {
public UseThread(String name) {
super(name);
}
@Override
public void run() {
String threadName = Thread.currentThread().getName();
System.out.println(threadName + " interrupt start flag =" + isInterrupted());
while(! isInterrupted()) {
try {
// The thread sleeps for 3 seconds
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
System.out.println("sleep error=" + e.getLocalizedMessage());
}
System.out.println(threadName + " is running");
System.out.println(threadName + " inner interrupt flag =" + isInterrupted());
}
System.out.println(threadName + " interrupt end flag =" + isInterrupted());
}
}
Copy the code
When we execute the previous method, the result is:
As you can see, the thread is still executing even if an exception is thrown, so be aware that throwing InterruptedException clears the interrupt flag bit, so let’s change the code to interrupt() again in a catch to interrupt the task
try {
// The thread sleeps for 3 seconds
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
// In the catch method, the task is interrupted by the interrupt() method.
interrupt();
System.out.println("sleep error=" + e.getLocalizedMessage());
}
Copy the code
Results:
After an exception is thrown, the thread is interrupted again in a catch, and the thread stops and no longer executes.
Sharing and collaboration between threads
3.1 Sharing between threads
As mentioned earlier, all threads of the same process share all the resources of that process. Sharing resources can cause A problem. When multiple threads access an object or A member variable of an object at the same time, it can cause data synchronization problems. After the operation is complete, the data is written to the memory. However, if the data has not been written to the memory, thread B also operates on the data and retrieves the data that has not been written to the memory. As a result, the data is not synchronized.
To address this problem, the keyword synchronized was introduced into Java (a separate article).
3.2 Collaboration between threads
Threads can cooperate with each other to complete A task. For example, when thread A changes A value, it needs to notify another thread to perform subsequent operations. The whole process starts with one thread and ends with another thread, the former being the producer and the latter the consumer.
In Android, when we make a network request, we often open a new subthread to make a network request to obtain data. After obtaining the data, we will notify the main thread to update UI through Handle. This operation reflects the cooperation between threads.
3.2.1, nitify(), notifyAll(), wait() wait/notification mechanisms
Thread A calls the wait() method of object O to enter the wait state, while thread B calls the notify() or notifyAll() method of object O. After receiving the notification, thread A returns from the wait() method of object O and performs subsequent operations. The two threads interact through object O, and the relationship between wait() and notify and notifyAll() on the object is like a switch signal to complete the interaction between the waiting and notifying parties.
We know that the Object class is the parent of all classes, and there are related methods in the Object class
Notify () :
Notify a thread WAITING on an object to return from wait if it has acquired the lock on the object. The thread that has not acquired the lock is WAITING again.
NotifyAll () :
Notifies all threads waiting on the object
wait()
The thread calling this method enters a WAITING state and returns only when it is notified by another thread or interrupted. Note that after calling wait(), the lock on the object is released
wait(long)
Timeout Wait time, in this case the parameter time is milliseconds, that is, wait for n milliseconds, if no notification timeout return
wait (long,int)
For more granular control of the timeout, it can be nanoseconds
Here is a case to illustrate: On Singles’ Day, you bought three items of goods, and you waited anxiously at home. If you had nothing to do, you would refresh your mobile phone to see the information of the goods delivered. We will simulate the update of the information of the goods delivered.
public class NwTest {
// Place of shipment
public String location = "Chongqing";
// All the goods will be on a different train. When the goods arrive at the next station, the corresponding express information will be updated respectively
public synchronized void changeLocationNotify(String location) {
this.location = location;
this.notify();
}
// All the goods are on the same express bus. When the goods arrive at the next station, all the information is updated.
public synchronized void changeLocationNotifyAll(String location) {
this.location = location;
System.out.println("changeLocationNotifyAll");
this.notifyAll();
}
public static class LocationThread extends Thread {
public final NwTest mNwTest;
public LocationThread(NwTest nwTest) {
this.mNwTest = nwTest;
}
@Override
public void run(a) {
super.run();
try {
synchronized (mNwTest) {
System.out.println("LocationThread current location : " + mNwTest.location);
// Wait for the location to update
mNwTest.wait();
String name = Thread.currentThread().getName();
// Get the merchant information of the current item
System.out.println(LocationThread -- >current thread name: + name);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
// Get the updated location
System.out.println("LocationThread update location : " + mNwTest.location);
}
}
Copy the code
Note:
Wait (), notifyAll(), and notify() can only be used in synchronized methods or synchronized blocks
Otherwise it will throw IllegalMonitorStateException anomalies
Call:
public static void main(String[] args) throws InterruptedException {
NwTest nwTest = new NwTest();
for (int x = 0; x < 3; x++) {
new LocationThread(nwTest).start();
}
// After three days of simulation
Thread.sleep(3000);
// Notify individual product information to be updated
nwTest.changeLocationNotify(Hefei "");
}
Copy the code
We started three threads to simulate the three items you bought, and if you use notify(), just update the information for a single item (random wake up).
Results:
We saw three goods shipped at the same time, among which Thread_0 arrived in Hefei first and updated the data.
If notifyAll() is used, all shipping information is refreshed.
public static void main(String[] args) throws InterruptedException {
NwTest nwTest = new NwTest();
for (int x = 0; x < 3; x++) {
new LocationThread(nwTest).start();
}
Thread.sleep(3000);
// Notify three items for information update
nwTest.changeLocationNotifyAll(Hefei "");
}
Copy the code
Results:
This is the basic use of notifyAll(), notify(), and wait(), where wait(long) means that the thread waits n milliseconds and automatically executes the subsequent methods if no notifyAll() or notify() is received.
Based on the Demo above, we can tidy up the standard paradigm for waiting and notification
wait():
1) Obtain the lock of the object.
2) Call the wait() method based on the judgment criteria.
3) If the conditions are met, the corresponding logic is executed.
Notify () or notifyAll ()
1) Obtain the lock of the object.
2) Change conditions and send notifications.
3) Notify all threads waiting on the object.
The above is mainly some basic concepts of sorting multithreading, as well as the basic use of notify() and wait(), the keyword synchronized prepared for the next separate sorting, subsequent planning sorting thread pool related knowledge and Android AsyncTask source code analysis, Please give me a thumbs up if you like!