线程的状态包括五种:开始、准备运行、运行、阻塞、死亡
其中,开始是指线程被创建之后,但没有start之前的状态。
准备运行是指线程调用start方法之后,等待线程规划器调用执行。
运行就是指正在运行的线程
阻塞是指正在运行的线程被阻塞
死亡就是调用stop方法或者run方法执行结束之后的线程消亡的状态。
首先先来看个表,如下:
状态
阻塞
可中断
描述
正运行
正在处理器上运行
准备运行
等待轮流获得处理器时间
休眠
可
可
经过一段时间,或者被中断后准备运行
等待
可
可
接到通知,或超时,或被中断后准备运行
阻塞于I/O
可
I/O条件改变后(例如读取了一个字节数据)准备运行
阻塞于同步
可
当获得锁后准备运行(传递同步语句)
线程是否运行由线程规划器来决定,线程规划器控制哪个准备运行的线程线程运行于处理器上,在任何时刻只有一个线程处于运行状态。
正在运行和准备运行就不多介绍了,下面介绍下阻塞状态。
线程可以阻塞于4种状态,分别为:
1、调用Thread.sleep()后,该线程将一直阻塞(休眠)到指定的毫秒时间段之后,或者被中断请求打断。
2、当线程碰到wait()语句时,该线程将一直阻塞到接到通知、被中断或者经过了指定的毫秒时间段为止(指定了超时值)。
3、线程被I/O请求阻塞,这种情况有很多种,它可以无限阻塞,因而不能指定超时。
4、因线程所需资源加锁,没有得到该资源而被阻塞。典型的是synchronized语句和sychronized方法,线程将一直阻塞于synchronized,直到获得了指定资源的锁为止。它可以无限阻塞,因而不能指定超时。
阻塞线程的条件改变之后,该线程就进入了准备运行状态,等待线程规划器选中运行之前,将一直保持这种状态。
新线程在调用start()方法后,才进去准备运行状态。
运行中的线程通过调用yield()方法或者被规划器交换出去而变成准备运行状态。这里提到了yield就简单的介绍下它吧。
yield方法与sleep()类似,只是不能由用户指定暂停多长时间,并且yield()方法只能让同优先级的线程有执行的机会。因此所有执行yield()的线程有可能在进入到可执行状态后马上又被执行,所以yield()方法只能使同优先级的线程有执行的机会。使用yield()的目的是让相同优先级的线程之间能适当的轮转执行。但是,实际中无法保证yield()达到让步目的,因为让步的线程还有可能被线程调度程序再次选中。yield()也不会释放锁标志。
实际上,yield()方法对应了如下操作;先检测当前是否有相同优先级的线程处于同可运行状态,如有,则把CPU的占有权交给次线程,否则继续运行原来的线程,所以yield()方法称为“退让”,它把运行机会让给了同等级的其他线程。
sleep和yield区别
sleep 方法允许较低优先级的线程获得运行机会,但yield()方法执行时,当前线程仍处在可运行状态,所以不可能让出较低优先级的线程此时获取CPU占有权。在一个运行系统中,如果较高优先级的线程没有调用sleep方法,也没有受到I/O阻塞,那么较低优先级线程只能等待所有较高优先级的线程运行结束,方可有机会运行。