Small knowledge, big challenge! This article is participating in the creation activity of “Essential Tips for Programmers”.

This article has participated in the “Digitalstar Project” and won a creative gift package to challenge the creative incentive money.

Brief introduction:

Thread is the smallest unit of operating system scheduling. In multi-core environment, multiple threads can be executed at the same time. If properly used, it can significantly improve program performance.

A preliminary understanding of threads

1. What is a thread

When the operating system runs a program, it starts a process for it. For example, starting a Java program creates a Java process. The smallest unit of modern operating system scheduling is thread, also known as Light Weight Process (Light Weight Process), a Process can create one to multiple threads, threads have their own counters, stacks and local variables and other attributes, and access to shared memory variables. The processor executes the program by quickly switching these threads.

Java itself is multithreaded

Sample code:

package com.lizba.p2; import java.lang.management.ManagementFactory; import java.lang.management.ThreadInfo; import java.lang.management.ThreadMXBean; import java.util.Arrays; /** * <p> * * </p> * * @Author: Liziba * @Date: 2021/6/13 23:03 */ public class MultiThread {public static void main(String[] args) {// Obtain the Java thread management MXBean threadMXBean = ManagementFactory.getThreadMXBean(); // Get thread and thread stack information; // Boolean lockedMonitors = false No need to obtain synchronized monitor information; // Boolean lockedsynchronizer = false Do not need synchronizer information threadMXBean.dumpAllThreads(false, false); Stream (threadInfos).foreach (threadInfo -> {system.out.println ("[" + threadinfo.getThreaDid () +) "]" + threadInfo.getThreadName()); }); }}Copy the code

Output results (not necessarily consistent) :

[6]Monitor ctrl-break

[5]Attach a Listener // Attach a thread of communication between JVM processes

[4]Signal Dispatcher // The thread that processes signals sent to the JVM

[3]Finalizer // Finalizer thread for calling the object

[2]Reference Handler // Clear the thread of Reference

[1] Main // Main thread, user program entry


Conclusion:

As you can see from the output, Java programs themselves are multithreaded. Instead of just one main thread running, the main thread is running at the same time as multiple other threads.

3. Why multithreading

The benefits of using multiple threads are as follows:

  1. More processor cores

As the number of computer processor cores increases, it develops from high frequency to multi-core technology. Now computers are better at parallel computing, so how to make full use of multi-core processor is now the main problem. Threads are the smallest unit of operating system scheduling. When a program is run as a process, it creates multiple threads, and a thread can only run on one processor at a time. Therefore, a process that can use multithreaded computation and divide its computation logic across multiple processor cores will have a significant performance improvement over a single thread.

  1. Faster response time

In complex business scenarios, we can dispatch non-consistently correlated business to other threads (or use message queues). This reduces the application response time to user requests

  1. A better programming model

Using the multithreaded programming model provided by Java allows programmers to better solve problems without having to think too hard about how to multithread them.

4. Thread priority

Modern operating system basically adopts the way of time slice allocation to schedule threads, that is, the operating system divides CPU operation into time slices, threads will allocate a number of time slices, when the thread time slice is used up, thread scheduling will wait for the next time slice allocation. How long a thread can execute in a CPU schedule depends on how many time slices are divided, and thread priority is a thread attribute that determines whether a thread needs to allocate more or less processor resources.

In Java threads, the priority of a thread can be set from 1 to 10, and the default priority is 5. In theory, threads with a higher priority should allocate more time slices than threads with a lower priority.


Sample code:

package com.lizba.p2; import java.util.ArrayList; import java.util.List; import java.util.concurrent.TimeUnit; / * * * < p > * thread priority setting * < / p > * * @ Author: Liziba * @ the Date: 2021/6/14 12:03 */ public class Priority {private static volatile Boolean notStart = true; */ private static volatile Boolean notEnd = true; public static void main(String[] args) throws InterruptedException { List<Job> jobs = new ArrayList<>(); For (int I = 0; int I = 0; i < 10; i++) { int priority = i < 5 ? Thread.MIN_PRIORITY : Thread.MAX_PRIORITY; Job job = new Job(priority); jobs.add(job); Thread thread = new Thread(job, "Thread:" + i); thread.setPriority(priority); thread.start(); } notStart = false; TimeUnit.SECONDS.sleep(10); notEnd = false; jobs.forEach( job -> System.out.println("Job priority : " + job.priority + ", Count : " + job.jobCount) ); */ static class Job implements Runnable {private int priority; private long jobCount; public Job(int priority) { this.priority = priority; } @override public void run() {while (notStart) {thread.yield (); } while (notEnd) {// Let the CPU time slice wait for the next scheduling thread.yield (); jobCount++; }}}}Copy the code

Execution Result:

The output shows that the number of times a priority 1 thread executes is very similar to the number of times a priority 10 thread executes, indicating that program correctness is not dependent on the priority of the thread.

\

5. Thread state

The lifecycle of a thread is as follows:

The name of the state instructions
NEW In the initial state, the thread is built without calling the start() method
RUNNABLE Running state, the Java thread refers to the ready and running states of the operating system collectively as running
BLOCKED The thread is blocked on the lock
WAITING A thread enters a wait state that indicates that the current thread is waiting for other threads to do some specific action (notification or interrupt)
TIME_WAITING WAITING can return within a specified time
TERMINATED Terminated status: indicates that the current thread has finished executing

Code to view the status of Java threads

Code examples:

package com.lizba.p2; import java.util.concurrent.TimeUnit; / * * * < p > work tools * * sleep time specified < / p > * * @ Author: Liziba * @ the Date: 2021/6/14 13:27 */ public class SleepUtil { public static final void sleepSecond(long seconds) { try { TimeUnit.SECONDS.sleep(seconds); } catch (InterruptedException e) { e.printStackTrace(); }}}Copy the code
package com.lizba.p2; / * * * < p > * thread state sample code * < / p > * * @ Author: Liziba * @ the Date: 2021/6/14 13:25 */ public class ThreadStateDemo { public static void main(String[] args) { // TimeWaiting new Thread(new  TimeWaiting(), "TimeWaitingThread").start(); // Waiting new Thread(new Waiting(), "WaitingThread").start(); New Thread(new Blocked(), "Blocked1Thread").start(); new Thread(new Blocked(), "Blocked2Thread").start(); Static class TimeWaiting implements Runnable {@override public void run() {while (true) {static class TimeWaiting implements Runnable {@override public void run() { SleepUtil.sleepSecond(100); Static class implements Runnable {@override public void run() {while (true) {static class implements Runnable {@override public void run() { synchronized (Waiting.class) { try { Waiting.class.wait(); } catch (InterruptedException e) { e.printStackTrace(); }}}}} // Lock the thread Blocked. Class instance, Static class Blocked implements Runnable {@override public void run() {synchronized (Blocked. Class) {while (true) { SleepUtil.sleepSecond(100); } } } } }Copy the code

View Java processes using JPS:

ThreadStateDemo process ID is 2576. Type jStack 2576 to see the output:

Collate the output results:

Name of the thread Thread state
Blocked2Thread BLOCKED (on object monitor), BLOCKED on the lock from which BLOCKED. Class was obtained
Blocked1Thread TIMED_WAITING (sleeping)
WaitingThread WAITING (on object monitor)
TimeWaitingThread TIMED_WAITING (sleeping)

Conclusion:

Threads are not mandated to be in one state during their life cycle, but switch between states as the code executes.

The Java thread state change diagram looks like this:

Java thread state transitions diagram

Conclusion:

  • After the thread is created, the start() method is called to start running
  • After executing wait(), a thread enters the wait state, which depends on other threads to return to the running state
  • The timeout waiting is equivalent to adding the timeout limit on the basis of waiting for Miss Zhu, and returning to the running state after reaching the set timeout time
  • When a thread executes a synchronized method or code block, the thread that does not acquire the lock will enter the blocking state.
  • The thread enters the terminating state after executing Runnable’s run() method
  • Threads blocking the Lock interface in Java’s Concurrent package are in a wait state because the implementation of the Lock interface blocking uses Daemon threads

Daemon threads

Brief introduction:

Daemon threads are support threads that perform background scheduling and support work. If no non-daemon threads exist in a Java VIRTUAL machine, the Java virtual machine will exit. Daemon threads need to be set before startup, not after.

Setting Mode:

Thread.setDaemon(true)
Copy the code

Points for special attention:

Daemon threads are used to complete support work, but the Finally block of Daemon threads does not necessarily execute when the Java VIRTUAL machine exits.

Sample code:

package com.lizba.p2; /** * <p> ** @author: Liziba * @date: 2021/6/14 19:50 */ public class DaemonRunner implements Runnable{ @Override public void run() { try { SleepUtil.sleepSecond(100); } finally { System.out.println("DaemonRunner finally run ..." ); }}}Copy the code

Testing:

package com.lizba.p2; /** * <p> * * </p> * * @Author: Liziba * @Date: 2021/6/14 19:59 */ public class DaemonTest { public static void main(String[] args) { Thread t = new Thread(new DaemonRunner(), "DaemonRunner"); t.setDaemon(true); t.start(); }}Copy the code

Output result:

Conclusion:

It is not hard to see that the finally block of DaemonRunner’s run method is not executed. This is because when the Java VIRTUAL machine runs out of non-Daemon threads, the virtual machine exits immediately and all Daemon threads in the virtual machine need to be terminated immediately. So the DaemonRunner thread is terminated immediately, and finally is not executed.

The thread starts and terminates

1. Construct threads

Before running a thread, you need to construct a thread object. When you construct a thread object, you need to set some thread properties, such as thread group, thread priority, whether it is a daemon thread, thread name, etc.

Code examples:

From Java. Lang. Thread

/** * Initializes a Thread. * * @param g the Thread group * @param target the object whose run() method gets called * @param name the name of the new Thread * @param stackSize the desired stack size for the new thread, or * zero to indicate that this parameter is to be ignored. * @param acc the AccessControlContext to inherit, or * AccessController.getContext() if null * @param inheritThreadLocals if {@code true}, inherit initial values for * inheritable thread-locals from the constructing thread */ private void init(ThreadGroup g, Runnable target, String name, long stackSize, AccessControlContext acc, boolean inheritThreadLocals) { if (name == null) { throw new NullPointerException("name cannot be null"); } // Set thread name this.name = name; Thread parent = currentThread(); SecurityManager security = System.getSecurityManager(); if (g == null) { if (security ! = null) { g = security.getThreadGroup(); } if (g == null) { g = parent.getThreadGroup(); } } g.checkAccess(); if (security ! = null) { if (isCCLOverridden(getClass())) { security.checkPermission(SUBCLASS_IMPLEMENTATION_PERMISSION); } } g.addUnstarted(); // set thread group this.group = g; // Set the daemon property to the parent thread's this.daemon = parent-isdaemon (); This.priority = parent.getPriority(); if (security == null || isCCLOverridden(parent.getClass())) this.contextClassLoader = parent.getContextClassLoader(); else this.contextClassLoader = parent.contextClassLoader; this.inheritedAccessControlContext = acc ! = null ? acc : AccessController.getContext(); this.target = target; setPriority(priority); / / copy InheritableThreadLocals attribute of the parent thread if (inheritThreadLocals && parent. InheritableThreadLocals! = null) this.inheritableThreadLocals = ThreadLocal.createInheritedMap(parent.inheritableThreadLocals); /* Stash the specified stack size in case the VM cares */ this.stackSize = stackSize; // Set a thread id tid = nextThreadID(); }Copy the code

Conclusion:

In the above code, a newly built thread object is allocated space by its parent thread, and the child inherits the parent Daemon, priority, contextClassLoader, and inheritable ThreadLocal. A unique ID is also assigned to identify the thread. At this point, a complete runnable thread object is initialized and waiting to run in heap memory.

2. What is a thread interrupt

Interrupts can be understood as an identifying bit property of a thread that indicates whether a running thread has been interrupted by another thread. A Thread responds by checking whether it has been interrupted, using isInterrupted(), or by calling the static thread.interrupted () method to restore the interrupt flag of the current Thread.

It is impossible to determine whether a thread has been interrupted in the following cases:

  1. The thread has been interrupted and isInterrupted() returns false even if it has been interrupted
  2. The InterruptedException method throws an exception. Calling isInterrupted() returns false even if interrupted because the interrupt flag is cleared before throwing InterruptedException.

Sample code:

package com.lizba.p2; / * * * < p > * thread interrupt example code * < / p > * * @ Author: Liziba * @ the Date: 2021/6/14 20:36 */ public class Interrupted {public static void main(String[] args) {// sleepThread sleepThread = new Thread(new SleepRunner(), "sleepThread"); sleepThread.setDaemon(true); // busyThread Thread busyThread = new Thread(new BusyRunner(), "busyThread"); busyThread.setDaemon(true); // Start two threads sleepthread.start (); busyThread.start(); // Sleep sleepThread and busyThread for 5 seconds. SleepSecond (5); // Interrupts two threads sleepthread.interrupt (); busyThread.interrupt(); System.out.println("SleepThread interrupted is " + sleepThread.isInterrupted()); System.out.println("BusyThread interrupted is " + busyThread.isInterrupted()); SleepSecond (2); // Sleeputil.sleepsecond (2); } static class SleepRunner implements Runnable { @Override public void run() { while (true) { SleepUtil.sleepSecond(10);  } } } static class BusyRunner implements Runnable { @Override public void run() { while (true) {} } } }Copy the code

View the run result:

Conclusion:

The sleepThread throws InterruptedException. The sleepThread’s interrupt flag returns false, even though both threads were interrupted. This is because timeunit.seconds.sleep (SECONDS) throws InterruptedException after the sleepThread’s interrupt flag is cleared. However, busyThread keeps running without throwing an exception, and the interrupt bit is not cleared.

Suspend (), resume(), and stop()

For example:

Thread these three methods, equivalent to QQ music play music when the suspension, recovery and stop operation. (Note that these methods are out of date and not recommended.)

Sample code:

package com.lizba.p2; import java.text.DateFormat; import java.text.SimpleDateFormat; import java.util.Date; import java.util.concurrent.TimeUnit; / * * * < p > * thread expired method example * < / p > * * @ Author: Liziba * @ the Date: 2021/6/14 20:57 */ public class Deprecated { static DateFormat format = new SimpleDateFormat("HH:mm:ss"); public static void main(String[] args) { Thread printThread = new Thread(new PrintThread(), "PrintThread"); printThread.start(); SleepUtil.sleepSecond(3); Printthread.suspend (); System.out.println("main suspend PrintThread at " + format.format(new Date())); SleepUtil.sleepSecond(3); Printthread.resume (); System.out.println("main resume PrintThread at " + format.format(new Date())); SleepUtil.sleepSecond(3); // Terminate printThread printthread.stop (); System.out.println("main stop PrintThread at " + format.format(new Date())); SleepUtil.sleepSecond(3); } static class PrintThread implements Runnable { @Override public void run() { while (true) { System.out.println(Thread.currentThread().getName() + "Run at " + format.format(new Date())); SleepUtil.sleepSecond(1); }}}}Copy the code

Output result:

Conclusion:

The output of the above code execution is consistent with the API instructions and our expectations, but the seemingly correct code hides many of these problems.

Existing problems:

  • The suspend() method is called without releasing occupied resources (such as locks), which can result in deadlocks
  • The stop() method does not guarantee the normal release of resources when terminating a thread, and may cause the program to be in an indeterminate working state

4. Terminate the thread correctly

  1. Call the interrupt() method of the thread
  2. Use a Boolean variable to control whether to stop the task and terminate the thread

Sample code:

package com.lizba.p2; / * * * < p > * sign a termination of the thread sample code * < / p > * * @ Author: Liziba * @ the Date: 2021/6/14 21:17 */ public class ShutDown { public static void main(String[] args) { Runner one = new Runner(); Thread t = new Thread(one, "CountThread"); t.start(); SleepUtil.sleepSecond(1); t.interrupt(); Runner two = new Runner(); t = new Thread(two, "CountThread"); t.start(); SleepUtil.sleepSecond(1); two.cancel(); } private static class Runner implements Runnable { private long i; private volatile boolean on = true; @Override public void run() { while (on && ! Thread.currentThread().isInterrupted()) { i++; } System.out.println("Count i = " +i); } /** * close */ public void cancel() {on = false; }}}Copy the code

Output result:

Conclusion:

The main thread terminates CountThread with an interrupt and cancel(). The advantage of both methods is that they give the thread a chance to clean up resources when it terminates. It’s safer and more elegant.

This article concludes with “The Art of Concurrent Programming in Java” and concludes with “Interthread Communication”.

\