线程被创建并启动后,它既不是一启动就进入执行状态,也不是一直处于执行状态,线程的生命周期包括:
新建(New)、就绪(Runnable)、运行(Running)、阻塞(Blocked)和死亡(Dead)
五种状态。线程启动以后,它不可能一直独自占着CPU,CPU需要在多条线程之间切换,所以线程的状态也会多次在运行,阻塞之间切换。
新建和就绪状态
当使用new关键字创建线程后,该线程就处于新建状态,此时他由java虚拟机为其分配内存,并初始化其成员变量的值,这时,程序并不会执行线程的线程执行体。
当线程对象调用了start()方法后,线程就会处于就绪状态,java虚拟机为其创建方法调用栈和程序计数器,处于这个状态中的程序并没有运行,只是表示该线程可以运行了。线程何时开始运行,取决于JVM里线程调度器的调度。
注意:启动线程使用start()方法,而不是run()方法!永远不要调用线程的run()方法!因为,调用start()方法,启动线程的时候,系统会把run()方法当成线程的执行体来处理;但是如果直接调用线程对象的run()方法,则run()方法会被立即执行,而且系统会把线程对象当成一个普通的对象,把run() 方法看成一个普通方法,而不是线程执行体。
注意:只能对处于新建状态的线程调用start()方法,否则将引发IllegalThreadStateException异常。
运行和阻塞状态
线程运行后,不可能一直处于运行状态(除非它的线程体足够的短,瞬间就执行结束了),线程在运行中需要被中断,目的是让其他线程获得执行的机会,线程调度的细节取决于底层平台所采用的策略。现代的桌面和服务操作系统都采用了抢占式调度策略(系统会给每个可执行的线程一个小时间段来处理任务,当这个时间段用完后,系统会剥夺该线程所占用的资源,让其他线程获得执行的机会),但是一些小型设备如手机可能采取的是协作式调度策略,在这样的系统中,只有当一个线程调用了它的sleep()或者yield()方法后,才会放弃所占用的系统资源——也就是必须由该线程主动放弃所占用资源。
线程状态转换图
上图可以看到,线程从阻塞状态只能进入就绪状态,无法直接进入运行状态。而就绪状态和运行状态间的转换通常不受程序的控制,而是由系统线程调度所决定的。
线程死亡
线程会以下面三种方式结束,结束后就处于死亡状态:
1、run()或者call()方法执行完成,线程正常结束
2、线程抛出一个未捕获的Exception或Error
3、直接调用该线程的stop()方法来结束该线程——这个方法容易导致死锁,通常不推荐使用
注意:当主线程结束时,其他线程并不受任何影响,并不会随之结束。一旦子线程启动起来后,它就具有和主线程一样的地位,不会受主线程的影响。
注意:不要试图对一个已死亡的线程调用start()方法使他重新启动,死亡就是死亡,这个线程不可再次作为线程执行。
下面尝试对处于死亡状态的线程再次调用start()方法:
public class StartDead extends Thread{
private int i;
//重写run()方法,run()方法的方法执行体就是线程执行体
public void run() {
for(;i<100;i++) {
System.out.println(currentThread().getName()+" "+i);
}
}
public static void main(String[] args) {
//创建线程对象
StartDead sd = new StartDead();
for(int i = 0;i<100;i++) {
System.out.println(Thread.currentThread().getName() + " " + i);
if(i==20) {
//启动线程
sd.start();
//判断启动后线程的isAlive()方法返回true
System.out.println(sd.isAlive());
}
//当线程处于新建、死亡两种状态时,isAlive()方法返回false
//当i>20时,该线程肯定已经启动过了,如果sd.isAlive()为假时
//那就是死亡状态了
if(i>20&&!sd.isAlive()) {
//试图再次启动该线程
sd.start();
}
}
}
}
运行结果如下(截取部分结果):
从上面结果,可以看到对已死亡的线程调用start()方法来再次启动线程时,会引发IllegalThreadStateException异常,这表明处于死亡状态的线程无法再次运行了。
注意:
1、不要对处于死亡状态的线程调用start()方法,程序只能对新建状态的线程调用start()方法
2、对新建状态的线程两次调用start()方法也是错误的
上面两种均会引发IllegalThreadStateException异常。
(参考《疯狂Java讲义第3版》)