Java后台线程_setDaemon ( true )语句&线程的生命周期及状态转换

120 阅读4分钟

本文已参与「新人创作礼」活动,一起开启掘金创作之路。

一、后台线程_setDaemon ( true )语句

对 Java 程序来说,只要还有一个前台线程在运行,这个进程就不会结束,如果一个进程中只有后台线程运行,这个进程就会结束

这里提到的前台线程和后台线程是一种相对的概念,新创建的线程默认都是前台线程,如果某个线程对象在启动之前调用 setDaemon ( true )语句,这个线程就变成一个后台线程

案例:

class DamonThread implements Runnable { //创建 DamonThread类,实现 Runnable接口
    public void run() { //实现接口中的 run()方发
        while (true) {
            System.out.println(Thread.currentThread().getName()+"---is running");
        }
    }
}

public class ExampleM06 {
    public static void main(String[] args) {
        System.out.println("main线程是后台线程吗?"+Thread.currentThread().isDaemon());
        DamonThread dt = new DamonThread(); //创建一个 DamonThread对象 dt
        Thread t = new Thread(dt,"后台线程");   //创建线程 t 共享 dt 资源
        System.out.println("t 线程默认是后台线程吗?"+t.isDaemon());
        //判断是否为后台线程
        t.setDaemon(true);  //将线程 t 设置为后台线程
        t.start();  //调用 start()方法开启线程 t
        for (int i=0;i<10;i++) {
            System.out.println(i);
        }
    }
}

当开启线程 t 后,会执行死循环中的打印语句,但我们将线程 t 设置为后台线程后,当前台线程死亡后, JVM 会通知后台线程。由于后台线程从接受指令,到做出响应,需要一定的时间,因此,打印了几次“后台线程---is running.”语句后,后台线程也结束了。由此说明进程中只有后台线程运行时,进程就会结束

注意:要将某个线程设置为后台线程,必须在该线程启动之前,也就是说 setDaemon()方法必须在 start()方法之前调用,否则会引发 IllegalThreadStateException 异常。

二、线程的生命周期及状态转换

  1. 新建状态(New) 创建一个线程对象后,该线程对象就处于新建状态,此时它不能运行,和其他Java象一样,仅仅由 Java 虚拟机为其分配了内存,没有表现出任何线程的动态特征。
  2. 就绪状态(Runnable) 当线程对象调用了 start()方法后,该线程就进人就绪状态(也称可运行状态)。处于就绪状态的线程位于可运行池中,此时它只是具备了运行的条件,能否获得 CPU 的使用权开始运行,还需要等待系统的调度
  3. 运行状态(Running) 如果处于就绪状态的线程获得了 CPU 的使用权,开始执行 run ()方法中的线程执行体,则该线程处于运行状态。当一个线程启动后,它不可能一直处于运行状态(除非它的线程执行体足够短,瞬间就结束了),当使用完系统分配的时间后,系统就会剥夺该线程占用的 CPU 资源,让其他线程获得执行的机会。需要注意的是,只有处于就绪状态的线程才可能转换到运行状态。
  4. 阻塞状态(Blocked) 一个正在执行的线在某些特殊情况下,如执行耗时的输入/输出操作时,会放弃 CPU 的使用权,进入阻塞状态。线程进人阻塞状态后,就不能进人排队队列。只有当引起阻塞的原因被消除后,线程才可以转人就绪状态。

下面就列举一下线程由运行状态转换成阻塞状态的原因,以及如何从阻塞状态转换成就绪状态。

1)当线程试图获取某个对象的同步锁时,如果该锁被其他线程所持有,则当前线程会进入阻寨状态,如果想从阻塞状态进入就绪状态必须得获取到其他线程所持有的锁。

2)当线程调用了一个阻塞式的IO方法时,该线程就会进人阻塞状态,如果想进人就绪状态就必须要等到这个阳塞的IO方法返回。

3)当线程调用了某个对象的 wait()方法时,也会使线程进人阻塞状态,如果想进人就绪状态就需要使用 notify ()方法唤醒该线程。

4)当线程调用了 Thread 的 sleep ( long millis )方法时,也会使线程进人阻塞状态,在这种情况下,只需等到线程睡眠的时间到了以后,线程就会自动进人就绪状态。

5)当在一个线程中调用了另一个线程的 join ()方法时,会使当前线程进入阻塞状态,在这种情况下,需要等到新加入的线程运行结束后才会结束阻塞状态,讲入就绪状态。

需要注意的是,线程从阻塞状态只能进人就绪状态,而不能直接进入运行状态,也就是说结束阻塞的线程需要重新进入可运行池中,等待系统的调度。

  1. 死亡状态(Terminated) 线程的 run ()方法正常执行完毕或者线程抛出一个未捕获的异常( Exception )、错误( Error ),线程就进人死亡状态。一旦进入死亡状态,线程将不再拥有运行的资格,也不能再转换到其他状态。