Whether you’re a freshman in college or a CRUD with 2-3 years of experience, that means you’ll be asked these questions for at least 3 years. Take the time to get to the bottom of them. The best way to get rid of your fear is to face him, Ollie! (This series is the notes and summary of my learning process, and provides debugging code for everyone to play with)

The chapter reviews

1. What are the main problems solved by concurrent programming?

2. How are threads created and implemented?

3. How does new Thread() wear three?

Please review the above questions by yourself, or the previous chapter if you have any questions

In this chapter the feed

By the end of this chapter, you will have a good understanding of the life cycle of a thread, how states transition from one state to another, and how to interpret Java thread state enumeration classes. In addition, this chapter will focus on the detailed interpretation and analysis of thread.start (), and you can understand the relationship between start() and run() more clearly through cases. (As always, if you are familiar with this section, you can choose to like 👍 directly to complete this chapter.)

This chapter code download

Thread life cycle

I believe that many students will not be angry to see these four words, the interview is always asked what is the life cycle, the life cycle you describe, a temper up. It is true that this thing is a bit annoying, but there is no personal skills, rote learning is the simplest and most useful way. Wait until memorized, naturally in the use of time will be integrated through, and even derived their own unique ideas. Don’t say too much, the conceptual thing back to finish the thing, serving ~

1. The NEW stage

The NEW phase is the phase when your NEW Thread() creates the Thread object.

To highlight the

In fact, the thread does not exist at all in the NEW stage, we just created a Therad object, and our most commonly used NEW keyword is the same truth. Our threads are created in the JVM process only when we actually start them up.

As we saw in the previous chapter, we need to call Thread.start() to start the Thread, which transitions from the new phase to the RUNNABLE phase.

2. The RUNNABLE phase

Only a call to thread.start () can cause a Thread to transition from the NEW phase to the RUNNABLE phase.

The thread must wait for the CPU to flip the switch before it can actually run. Some of you might say what if the CPU doesn’t change the brand? Technically speaking, a thread in a RUNNABLE has only two ways to exit accidentally, or it can be branded RUNNING by the CPU.


Up to this point, everything seems pretty simple and nice. New a Thread object, call start, and wait for the CPU to turn over the brand to enter the RUNNING phase. For example, in the NEW phase, our thread is still a beautiful object outside the palace. After calling the start method, it suddenly becomes a small master in the palace, namely the intermediate stage RUNNABLE. When it obtains the CPU scheduling execution right, it is promoted to the favorite goddess, namely entering the RUNNING phase. It is conceivable that at this time our thread lady must be more complicated than the beautiful woman outside the palace.


3. The RUNNING stage

⚠ ️ note

For those of you who have seen this, you might wonder, well, Java thread states don’t have this state, so why do we separate out this state when we talk about the life cycle? For the sake of smoothness, this explanation will be left to the next section, where we will continue with the thread life cycle.


Okay, let’s move on.

The thread in this phase has acquired CPU scheduling execution, that is, is in the running state.

In this phase, threads can transition forward or backward:

1. The thread enters the RUNNABLE phase because the CPU’s scheduler polling causes it to abort execution.

2. The thread initiates the RUNNABLE phase by invoking yield and relinquishing CPU execution. (This method does not work 100% and does not work when CPU resources are not strained.)

3. Call the sleep and wait methods to enter the BLOCKED stage (the BLOCKED stage should be distinguished from the BLOCKED state of the thread, and the general BLOCKED stage is described here).

4. The VM enters the BLOCKED stage after performing a BLOCKED I/O operation

5. The lock enters the BLOCKED phase by joining the lock to the blocking queue in order to obtain a lock resource

6. The thread ends or calls the stop method or tests a logical id to enter the TERMINATED phase

4. BLOCKED phase

The reason for entering this phase was explained in the RUNNING phase and will not be explained here. Instead, we will focus on how threads in this phase can be switched. 1. Go straight to TERMINATED, such as calling the stop method or unexpectedly dying (JVM Crash).

2. The thread blocks, reads or writes data, and enters the RUNNABLE state

3. The thread completes the sleep period and enters the RUNNABLE state

4. A thread in Wait state is awakened by notify or NotifyAll and enters the RUNNABLE state

5. Lock resources are obtained and the system enters the RUNNABLE state

6. The thread blocking process is interrupted, for example by calling the interrupt method and entering the RUNNABLE state

5. TERMINATED status

TERMINATED state is the final state of a thread. It does not switch to any other state. A thread in TERMINATED state means that its entire life is over.

1. The thread ends normally

2. The thread ends unexpectedly

3.JVM Crash

How the thread life cycle and Java thread state correspond (source code analysis)

The enumeration class java.lang.Thread.State is the enumeration class.

public enum State {
    NEW,
    RUNNABLE,
    BLOCKED,
    WAITING,
    TIMED_WAITING,
    TERMINATED;
}
Copy the code

As you can see here, the state enumeration class contains six thread states, which we read one by one.

(1) NEW state = NEW stage

     * <li>{@link #NEW}<br>
     *     A thread that has not yet started is in this state.
     *     </li>
Copy the code

The source code clearly states that a NEW state is the state in which a thread has just been created, but has not yet been started. This corresponds to the NEW phase in the previous section, which is not discussed here.

(2) RUNNABLE state = RUNNABLE +RUNNING state

 * <li>{@link #RUNNABLE}<br>
     *     A thread executing in the Java virtual machine is in this state.
     *     </li>
Copy the code

Threads executing in the Java virtual machine are in a state called RUNNABLE. That is, the RUNNABLE phase and the RUNNING phase that we talked about separately in the last section are called RUNNABLE states uniformly in thread states. The reason why we separate these two states when dividing the life cycle is that they are quite different. Threads in the RUNNING state are more complex than threads in the RUNNABLE state.

(3) BLOCKED +WAITING +TIMED_WAITING = BLOCKED

 * <li>{@link #BLOCKED}<br>
     *     A thread that is blocked waiting for a monitor lock
     *     is in this state.
     *     </li>
     * <li>{@link #WAITING}<br>
     *     A thread that is waiting indefinitely for another thread to
     *     perform a particular action is in this state.
     *     </li>
     * <li>{@link #TIMED_WAITING}<br>
     *     A thread that is waiting for another thread to perform an action
     *     for up to a specified waiting time is in this state.
     *     </li>
Copy the code

These three states are described in the source code like this

BLOCKED: The state in which a thread enters a blocking queue while waiting to acquire a monitor lock

WAITING: a thread in this state that is WAITING indefinitely for another thread to perform a specific operation.

TIMED_WAITING: a thread waiting for another thread to perform an operation is in this state for the specified waiting time.

Students are very interested inBLOCKEDandWAITINGIs the distinction between these two states clear? Clear straight into (4) Oh ~

Again, we enter the topic through the description of the state enumeration value in the source code. The BLOCKED state in state is described as follows:

 /**
         * Thread state for a thread blocked waiting for a monitor lock.
         * A thread in the blocked state is waiting for a monitor lock
         * to enter a synchronized block/method or
         * reenter a synchronized block/method after calling
         * {@link Object#wait() Object.wait}.
         */
        BLOCKED
Copy the code

The source code describes the state of BLOCKED as being BLOCKED while waiting to acquire a Monitor lock. In plain English, a synchronized block or method is described by the synchronized keyword. When another thread attempts to acquire the described block or method, the thread enters the BLOCKED state while waiting to acquire the Monitor lock. The key here is monitor Lock monitoring locks.

The wake up method is that the target monitor lock is released actively. At this time, we compete for the monitor lock and succeed.

WAITING state is described as follows:

/**
         * Thread state for a waiting thread.
         * A thread is in the waiting state due to calling one of the
         * following methods:
         * <ul>
         *   <li>{@link Object#wait() Object.wait} with no timeout</li>
         *   <li>{@link #join() Thread.join} with no timeout</li>
         *   <li>{@link LockSupport#park() LockSupport.park}</li>
         * </ul>
         *
         * <p>A thread in the waiting state is waiting for another thread to
         * perform a particular action.
         *
         * For example, a thread that has called <tt>Object.wait()</tt>
         * on an object is waiting for another thread to call
         * <tt>Object.notify()</tt> or <tt>Object.notifyAll()</tt> on
         * that object. A thread that has called <tt>Thread.join()</tt>
         * is waiting for a specified thread to terminate.
         */
        WAITING
Copy the code

Wait, Thread.join and locksupport. park will be in WAITING state respectively. There is no specific description of the Thread API here. There will be a separate chapter to introduce it, so we don’t have to worry about it.

WAITING state is the state in which a thread is WAITING for another thread to complete a particular operation. The key here is two threads. The target thread is WAITING for the other thread to complete an action.

To wake up a thread in WAITING state, call notify or notiffyAll after the other thread completes its logic.

(4) state TERMINATED = state TERMINATED


By now you should be familiar with the thread lifecycle and transitions between phases, as well as the Java thread state corresponding to the lifecycle of each phase. Next is the second section of this chapter source analysis ~

Delimit the key point, see tired students rest, next is start() source analysis

Thread. Start () design pattern source code interpretation

Before we start, why does start() start our thread, but not run()?

Nonsense do not say, first on the source code

/**
     * Causes this thread to begin execution; the Java Virtual Machine
     * calls the <code>run</code> method of this thread.
     * <p>
     * The result is that two threads are running concurrently: the
     * current thread (which returns from the call to the
     * <code>start</code> method) and the other thread (which executes its
     * <code>run</code> method).
     * <p>
     * It is never legal to start a thread more than once.
     * In particular, a thread may not be restarted once it has completed
     * execution.
     *
     * @exception  IllegalThreadStateException  if the thread was already
     *               started.
     * @see        #run()
     * @see        #stop()
     */
public synchronized void start() {
        /**
         * This method is not invoked for the main method thread or "system"
         * group threads created/set up by the VM. Any new functionality added
         * to this method in the future may have to also be added to the VM.
         *
         * A zero status value corresponds to state "NEW". * /if(threadStatus ! = 0) throw new IllegalThreadStateException(); /* Notify the group that this thread is about to be started * so that it can be added to the group's list of threads * and the group's unstarted count can be decremented. */
        group.add(this);

        boolean started = false;
        try {
            start0();
            started = true;
        } finally {
            try {
                if(! started) { group.threadStartFailed(this); } } catch (Throwable ignore) { /*do nothing. If start0 threw a Throwable then
                  it will be passed up the call stack */
            }
        }
    }
Copy the code

Well, we’ll start with the method description

/**
     * Causes this thread to begin execution; the Java Virtual Machine
     * calls the <code>run</code> method of this thread.
     * <p>
     * The result is that two threads are running concurrently: the
     * current thread (which returns from the call to the
     * <code>start</code> method) and the other thread (which executes its
     * <code>run</code> method).
     * <p>
     * It is never legal to start a thread more than once.
     * In particular, a thread may not be restarted once it has completed
     * execution.
     *
     * @exception  IllegalThreadStateException  if the thread was already
     *               started.
     * @see        #run()
     * @see        #stop()* /Copy the code

The start() method causes the thread to start running, and the Java virtual machine calls the run() method to execute the thread’s logical unit. And only allowed to start a thread, starting a thread is not legal for many times, throws IllegalThreadStateException anomalies.

Thread.start()

(1) Determine the threadStatus value

 /**
         * This method is not invoked for the main method thread or "system"
         * group threads created/set up by the VM. Any new functionality added
         * to this method in the future may have to also be added to the VM.
         *
         * A zero status value corresponds to state "NEW". * /if(threadStatus ! = 0) throw new IllegalThreadStateException();Copy the code

ThreadStatus ==0; threadStatus==0; threadStatus==0; = 0 so said thread is no longer in the NEW state, call this method at this time is illegal, so throw IllegalThreadStateException anomalies. (Such a simple logical judgment, I believe that the students and I must give birth to an idea “I can also 😁”)

(2) Join a thread group

       /* Notify the group that this thread is about to be started
         * so that it can be added to the group's list of threads * and the group's unstarted count can be decremented. */
        group.add(this);
Copy the code

Notify the group that a new thread has been started and needs to be added to the thread group, while reducing the count of unstarted threads. After reading the translation and having no idea, we continue to track Add

void add(Thread t) {
        synchronized (this) {
            if (destroyed) {
                throw new IllegalThreadStateException();
            }
            if (threads == null) {
                threads = new Thread[4];
            } else if (nthreads == threads.length) {
                threads = Arrays.copyOf(threads, nthreads * 2);
            }
            threads[nthreads] = t;

            // This is done last so it doesn't matter in case the // thread is killed nthreads++; // The thread is now a fully fledged member of the group, even // though it may, or may not, have been started yet. It will prevent // the group from being destroyed so the unstarted Threads count is // decremented. nUnstartedThreads--; }}Copy the code

The Add method is also divided into three small steps

Step 1: to determine whether a thread group has been destroyed, gone if the thread group, then throw IllegalThreadStateException anomalies.

 if (destroyed) {
                throw new IllegalThreadStateException();
            }
Copy the code

Step 2: Add the thread to the thread group array.

if (threads == null) {
                threads = new Thread[4];
            } else if (nthreads == threads.length) {
                threads = Arrays.copyOf(threads, nthreads * 2);
            }
            threads[nthreads] = t;

            // This is done last so it doesn't matter in case the // thread is killed nthreads++;Copy the code

If threads is null, create an array with an initial length of 4. Nthreads == threads.length if true, Threads = array.copyof (threads, nThreads * 2) add the input parameter to threads[] and add the subscript nthreads++

Step 3: The count of unstarted threads is reduced by one

// The thread is now a fully fledged member of the group, even
            // though it may, or may not, have been started yet. It will prevent
            // the group from being destroyed so the unstarted Threads count is
            // decremented.
            nUnstartedThreads--;
Copy the code

(3) Call start0() to start the thread

 boolean started = false;
        try {
            start0();
            started = true;
        } finally {
            try {
                if(! started) { group.threadStartFailed(this); } } catch (Throwable ignore) { /*do nothing. If start0 threw a Throwable then
                  it will be passed up the call stack */
            }
        }
        
Copy the code

Started is the starting state of the thread. The default value is false and assigned to true after starting. The start0() method is then called

    private native void start0();
Copy the code

Recall that our thread execution unit is written in run(), and there is nowhere to call run() except the start0() method. You can see that this method is a local method with a static block at the beginning of the Thread. Java class definition to complete the registration.

 /* Make sure registerNatives is the first thing <clinit> does. */
    private static native void registerNatives();
    static {
        registerNatives();
    }
Copy the code

Okay, so we’ll stop there, and then we’ll cover some of the underlying stuff in the JVM, but we won’t extend it here, just know that the start() internal call calls the start0() method, and ultimately calls the run() method inside the thread after we’ve created our thread.

Just keep in mind that the JVM actually creates the thread and calls the run method, and we’re just doing the act of sending a thread start signal as a startup initiator. We’re not going to take you around a bunch of CPP files anymore to avoid motion sickness.

This should clear up the problem at the beginning of this section, why does start() start our thread, but not run()? I’m sure you already know why, because run is only an internal implementation class, and if we simply call run, we just implement the logic in our own thread without actually opening up a new thread to execute our logic. Starting a new thread requires a call to start0() to get the JVM to do it for us, so only a call to the start() method can start the thread.

Here I also write a small example to verify:

 public static class TestStartAndRun implements Runnable{

    @Override
    public void run() {

      System.out.println("My name is."+Thread.currentThread().getName());
    }
  }


  public static void main(String[] args) {

    new Thread(new TestStartAndRun(),"start").start();

    new Thread(new TestStartAndRun(),"run").run();

  }
Copy the code

Output:

My name is Start. My name is MainCopy the code

As you can see, the Thread named run is not started. Thread.run() is executed by our main Thread. This proves that run cannot create a Thread, but can execute the logical unit of the inner class run multiple times in the current Thread.

4. Expand reading – Write a similar program that mimics the start template

/ / final void Teacup(String nothing){system.out.print ("Today");
    Brewing(nothing);
    System.out.println("Happy day begins!"); } @param nothing */ void Brewing(String nothing){} public static void main(String[] args) { TemplateDesignExample t1 = newTemplateDesignExample(){
      @Override
      void Brewing(String nothing){
        System.out.print("Rain,"+nothing); }}; t1.Teacup("Coffee in the morning.");


    TemplateDesignExample t2 = new TemplateDesignExample(){
      @Override
      void Brewing(String nothing){
        System.out.print("Sunny days,"+nothing); }}; t2.Teacup("Tea in the morning");
  }
Copy the code

Output:

It's raining today, coffee in the morning, happy day begins! It is sunny today, drink tea in the morning, and a happy day begins!Copy the code

So whether it is raining or sunny, we can drink what we want to drink, every day can happily code, if you can little attention, little praise 👍, every day is double happiness oh!

I wish the students happy Dragon Boat Festival holiday, holiday also don’t forget to study with me oh ~