这个问题出自阿里p6岗位第一面的提问,你会回答吗?
在Java里面,一个线程只能调用一次start()方法,第二次调用会抛出IllegalThreadStateException异常。
一个线程本身是具备一个生命周期的。
在Java里面,线程的生命周期包括6种状态。
- 新建(NEW),表示线程被创建出来还没真正启动的状态,可以认为它是个Java内部状态。
- 就绪(RUNNABLE),表示该线程已经在JVM中执行,当然由于执行需要计算资源,它可能是正在运行,也可能还在等待系统分配给它CPU片段,在就绪队列里面排队。
- 阻塞(BLOCKED),这个状态和我们前面两讲介绍的同步非常相关,阻塞表示线程在等待Monitor lock。比如,线程试图通过synchronized去获取某个锁,但是其他线程已经独占了,那么当前线程就会处于阻塞状态。
- 等待(WAITING),表示正在等待其他线程采取某些操作。一个常见的场景是类似生产者消费者模式,发现任务条件尚未满足,就让当前消费者线程等待(wait),另外的生产者线程去准备任务数据,然后通过类似notify等动作,通知消费线程可以继续工作了。Thread.join()也会令线程进入等待状态。
- 计时等待(TIMED_WAIT),其进入条件和等待状态类似,但是调用的是存在超时条件的方法,比如wait或join等方法的指定超时版本,如下面示例:
public final native void wait(long timeout) throws InterruptedException;
- 终止(TERMINATED),不管是意外退出还是正常执行结束,线程已经完成使命,终止运行,也有人把这个状态叫作死亡。
当我们第一次调用start()方法的时候,线程的状态可能处于终止或者非NEW状态下的其他状态。
再调用一次start(),相当于让这个正在运行的线程重新运行,不管从线程的安全性角度,还是从线程本身的执行逻辑,都是不合理的。
因此为了避免这个问题,在线程运行的时候会先判断当前线程的运行状态。
通过下面的例子来理解它:
package com.yiibai;
public class TestThreadTwice1 extends Thread {
public void run() {
System.out.println("Start running...");
}
public static void main(String args[]) {
TestThreadTwice1 t1 = new TestThreadTwice1();
t1.start();
t1.start();
}
}
执行上面示例代码,将会抛出异常:
Exception in thread "main" Start running...
java.lang.IllegalThreadStateException
at java.lang.Thread.start(Unknown Source)
at com.yiibai.TestThreadTwice1.main(TestThreadTwice1.java:11)