1、线程的生命周期
-
线程的new状态:当我们用关键字new创建一个thread对象时,此时它并不处于执行状态,因为没有调用start方法启动线程,那么线程的状态为new状态
在该状态中,线程的状态可以发生如下的状态转换:
-进入runnable状态:new状态通过start方法进入runnable状态。 -
线程的runnable状态:new状态通过start方法进入runnable状态,线程的运行与否和进程一样都要听令于cup的调度,我们把这个中间状态称为可执行状态(runnable),也就是说它具备执行的资格,但是并没有真正地执行起来而是在等待cpu的调度。
在该状态中,线程的状态可以发生如下的状态转换:
-进入running状态:runnable线程获得cpu的调度执行权进入running状态
-进入terminated状态:runnable线程意外终止进入terminated状态 -
线程的running状态:一旦cpu通过轮询或者其他方式从任何可执行队列中选中了线程,那么此时它才能真正地执行自己的逻辑代码。
在该状态中,线程的状态可以发生如下的状态转换:
-1.进入terminated状态:调用JDK已经不推荐使用的stop方法或者判断某个逻辑标识
-2.进入blocked状态:调用sleep或者wait方法而加入了waitSet中。进行某个阻塞的IO操作,比如因网络数据的读写而进入blocked状态。获取某个锁资源从而加入到该锁的阻塞队列中而进入了blocked状态。
-3.进入runnable状态:由于cup的调度器轮询使该线程放弃执行,进入runnable状态。线程主动调用yield方法,放弃cup执行权,进入runnable状态。 -
线程的blocked状态
在该状态中,线程的状态可以发生如下的状态转换:
-1.进入terminated状态:JVM崩溃,线程意外死亡。调用JDK不推荐使用的stop方法
-2.进入runnable状态:线程阻塞操作结束。线程完成了指定时间的休眠。Wait中的线程被其他线程notify/notifyall唤醒。线程获取到某个资源的锁,线程在阻塞过程中被打断,比如其他线程调用了interrupt方法。 -
线程的terminated状态:是一个线程的最终状态,在该状态中线程将不会切换到其他任何状态,线程进入terminated状态,意味着该线程的整个生命周期都结束了。
下列情况线程会进入terminated状态:
-1.线程运行正常结束,结束生命周期
-2.线程运行出错,意外结束
-3.JVM 崩溃,导致所有线程都结束
2、代码演示
/**
* 线程状态
* */
public class ThreadStatus implements Runnable{
// @Override
// public void run() {
// //(blocked)状态
// while(true) {
// System.out.println("自定义的线程执行了。。。");
// try {
// Thread.sleep(100);//线程处于(blocked)状态,当100毫秒过去之后,就会自动进入runnable状态
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
// }
// }
@Override
public synchronized void run() {
//(blocked)状态
while(true) {
try {
wait();//线程处于(blocked)状态
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("自定义的线程执行了。。。");
}
}
public static void main(String[] args) {
// 初始化状态
ThreadStatus n = new ThreadStatus();
Thread thread = new Thread(n);//创建线程并指定线程任务
thread.start();//启动线程
while(true) {
synchronized(n) {
System.out.println("主线程执行了。。。");
try {
Thread.sleep(100);//线程处于(blocked)态,当100毫秒过去之后,就会自动进入就runnable
} catch (InterruptedException e) {
e.printStackTrace();
}
n.notifyAll();//唤醒所有等待状态的线程
}
}
}
}