1.3 多线程 - 线程的生命周期

2018-07-15  本文已影响15人  Hey_Shaw
线程状态转换图

当前线程启动以后,不可能一直“霸占” CPU 独立运行,所以 CPU 需要在多条线程之间切换,于是线程状态也会多次在运行、阻塞之间切换。

新建与就绪状态

当程序使用 new 关键字创建了一个线程之后,该程序处于新建状态,此时和其他的 Java 对象一样,仅仅由 Java 虚拟机为其分配内存,并初始化其成员变量的值。此时的线程对象没有表现出任何线程的动态特征,程序也不会执行线程的线程执行体。

当线程对象调用 start() 方法之后,该线程处于就绪状态,Java 虚拟机会为其创建方法调用栈和程序计数器,处于这个状态中的线程并没有开始运行,只是表示该线程可以运行了。至于该线程何时开始运行,取决于 JVM 里线程调度器的调度。

启动线程使用 start() 方法,而不是 run() 方法!调用 start() 方法来启动线程,系统会把该 run() 方法当成线程执行体来处理;但如果直接调用线程对象的 run() 方法,则 run() 方法立即就会被执行,而且在 run() 方法返回其他线程无法并发执行 —— 也就是说,如果直接调用线程对象的 run() 方法,系统把线程对象当成一个普通对象,而 run() 方法也是一个普通方法,而不是线程执行体。且不能直接通过 getName() 获取当前执行线程的名字,而是需要使用 Thread.currentThread() 方法先获取当前线程,再调用线程对象的 getName() 方法获取线程的名字。

start() 方法的执行过程

Thread源码本身的 run() 方法:

private Runnable target;

@Override
public void run() {
      if (target != null) {
            target.run();
      }
}

target参数在Thread构造函数中被赋值,如果执行的线程为继承的 Thread,则线程启动执行子类覆盖的 run() 方法体,如果是实现的 Runnable 接口,执行的则是实现类的 run() 方法;看代码:

public class FirstThread extends Thread {
    
    private int i;
    
    public FirstThread() {} 
    
    public FirstThread(Runnable r) {
        super(r) ;
    }
    
    // 重写run方法,run方法的方法体就是线程执行体
    public void run() {
        super.run();    // 同时执行 Thread 的run() 方法才会执行 Runnable 接口实现类的run() 方法
        for (; i < 5; i++) {
            System.err.println("FirstThread:"+getName() + " " + i);
        }
    }

    public static void main(String[] args) {
        for (int i = 0; i < 5; i++) {
            System.out.println(Thread.currentThread().getName() + " " + i);
            if (i == 2) {
                new FirstThread().start();
                new FirstThread(new SecondThread()).start(); // 如果子类run() 方法未有super.run(),则Runnable接口不会执行
                new Thread(new SecondThread()).start();
            }
        }
    }
}

具体内容可参考:Java 中的进程与线程

调用线程对象的 start() 方法之后,该线程立即进入就绪状态 —— 即“等待状态”,但该线程并未真正进入运行状态。只能对处于新建状态的线程调用 start() 方法,否则将引发 IllegalThreadStateException 异常。

运行和阻塞状态

发生如下情况时,线程将会进入阻塞状态

发生如下情况时,线程将会进入就绪状态

线程状态转换图

从图可知,线程从阻塞状态只能进入就绪状态,无法直接进入运行状态。而就绪和运行状态之间的转换通常不受程序控制,而是由系统线程调度所决定,当处于就绪状态的线程获得处理器资源时,该线程进入运行状态;当处于运行状态的线程失去处理器资源时,该线程进入就绪状态。但有一个方法例外,调用 yield() 方法可以让运行状态的线程转入就绪状态。

线程死亡

线程会以如下三种方式结束,结束后就处于死亡状态

当主线程结束时,其他线程不受任何影响,并不会随之结束。一旦子线程启动起来后,它就拥有和主线程相同的地位,它不会受主线程的影响。

测试某个线程是否已经死亡,可以调用线程对象的 isAlive() 方法,当线程处于就绪、运行、阻塞三种状态时,该方法将返回 true;当线程处于新建、死亡两种状态时,该方法将返回 false。

Java中可以通过 interrupt() 中断线程,并通过 interrupted() 方法来判断线程是否已经中断,来结束线程 run() 执行体,从而终止当前线程。

上一篇 下一篇

猜你喜欢

热点阅读