线程状态
线程状态值可以通过Thread.getState()调用获取得到一个 java.lang.Thread.State 的枚举类型(Enum),定义了如下几种情况(均为大写):
- New:线程已创建而未启动,一个实例线程只能被启动一次,即一个线程只能一次处于该状态;
- Runnable:可以看成复合状态,包括两个子状态:
- Ready:表示该线程可以被线程调度器进行调度而使之处于Running状态(又叫活跃线程);
- Running:表示该线程run方法正在执行,执行 Thread.yield() 的线程,相应状态可能会由Running转为Ready。
- Ready:表示该线程可以被线程调度器进行调度而使之处于Running状态(又叫活跃线程);
- Blocked:一个线程发起一个阻塞式 I/O(Blocking I/O) 操作后,或者申请一个由其它线程持有的独占资源(比如锁)时,相应的线程会处于该状态。处于Blocked状态的线程并不占用处理器资源,当阻塞式I/O操作完成后或者线程获得了其申请的资源,该线程的状态又可以转换为Runnable。
- Waiting:执行了特定方法就会处于这种等待其他线程执行另外一些特定操作的状态:
- Object.wait()
- Thread.join()
- LockSupport.park(Object)
- 由waiting转为Runnable状态的方法:
- Object.notify()/notifyAll()
- LockSupport.unpack(Object)
- Object.notify()/notifyAll()
- Object.wait()
- Time_waiting:类似Waiting,差别在于该状态的线程并非无限制的等待其他线程执行特定操作,而是处于带时间限制的等待状态。当其他线程没有在指定时间内执行该线程所期望的特定操作时,该线程状态自动转为Runnable。下列为带超时的方式:
- Thread.sleep()
- Object.wait()
- Thread.join()
- LockSupport.parkNanos()
- LockSupport.parkUntil()
- Thread.sleep()
- Terminated:已经执行结束的线程处于该状态。一个线程只能处于一次该状态,Thread.run()正常返回或抛出异常提前终止都会处于该状态。
线程状态流转
- 初始线程处于 NEW 状态,调用 start() 之后开始执行,进入 RUNNING 或者 READY 状态。如果没有调用任何的阻塞函数,线程只会在 RUNNING 和 READY 之间切换,也就是系统的时间片调度。这两种状态的切换是操作系统完成的,开发者基本没有机会介入,除了可以调用 yield() 函数,放弃对 CPU 的占用。
- 一旦调用了图中的任何阻塞函数,线程就会进入 WAITING 或者 TIMED_WAITING 状态,两者的区别只是前者为无限期阻塞,后者则传入了一个时间参数,阻塞一个有限的时间。如果使用了 synchronized 关键字或者 synchronized 块,则会进入 BLOCKED 状态。
- 除了常用的阻塞/唤醒函数,还有一对不太常见的阻塞/唤醒函数,LockSupport.park()/unpark()。这对函数非常关键,Concurrent 包中 Lock 的实现即依赖这一对操作原语。故而 t.interrupted() 的精确含义是“唤醒轻量级阻塞”,而不是字面意思“中断一个线程”。
轻量级阻塞和重量级阻塞:
- 能够被中断的阻塞称为轻量级阻塞,对应的线程状态是WAITING或者TIMED_WAITING
- synchronized 这种不能被中断的阻塞称为重量级阻塞,对应的状态是BLOCKED
// pom.xml配置
<dependency>
<!-- 单元测试 -->
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.13</version>
<scope>compiler</scope>
</dependency>
// 测试
public class StateTest {
/**
* 输出当前的线程信息
*
* @param thread
*/
private static void print(Thread thread, String state) {
final StringBuilder builder = new StringBuilder();
builder.append(state)
.append("\n\tThread Name: ")
.append(thread.getName())
.append("\n\t")
.append("Thread State: ")
.append(thread.getState())
.append("\n");
System.out.println(builder.toString());
}
@Test
public void testThreadState() throws InterruptedException {
// 状态切换:新建->运行->终止
Runnable runnable = () -> {
print(Thread.currentThread(), "运行时:");
};
final Thread thread = new Thread(runnable, "测试线程");
// 输出刚创建Thread,没调用start方法的状态
print(thread, "未调用start方法:");
// 线程运行时的状态
thread.start();
// 线程结束后的状态
// 主线程休眠2s,确保测试线程结束
TimeUnit.SECONDS.sleep(2);
print(thread, "线程结束后:");
}
@Test
public void testThreadTimeSleepState() throws InterruptedException {
// 状态切换:新建->运行->等待->运行->终止
Runnable runnable = () -> {
print(Thread.currentThread(), "休眠之前:");
try {
// 当前线程休眠2s
TimeUnit.SECONDS.sleep(2);
} catch (InterruptedException e) {
e.printStackTrace();
}
print(Thread.currentThread(), "休眠之后:");
};
final Thread thread = new Thread(runnable, "测试线程");
// 输出刚创建Thread,没调用start方法的状态
print(thread, "未调用start方法:");
// 线程运行时的状态
thread.start();
// 线程休眠状态
// 主线程休眠1s,确保测试线程进入休眠
TimeUnit.SECONDS.sleep(1);
print(thread, "主线程休眠1s的状态:");
// 主线程休眠4s,确保测试线程结束
TimeUnit.SECONDS.sleep(4);
print(thread, "主线程休眠4s后的线程状态:");
}
@Test
public void testThreadBlockingState() throws InterruptedException {
// 线程状态:新建->运行->阻塞->运行—>终止
Runnable runnable = () -> {
// 对当前对象加锁
synchronized (this) {
print(Thread.currentThread(), "获得锁:");
}
};
final Thread thread = new Thread(runnable, "测试线程");
// 对当前对象加锁,这样thread将无法获取锁
synchronized (this) {
print(thread, "测试线程未启动的状态:");
// 启动测试线程,会被阻塞
thread.start();
// 获取thread线程状态
print(thread, "测试线程刚启动状态:");
// 等待2s,再看状态
TimeUnit.SECONDS.sleep(2);
print(thread, "测试线程启动后2s:");
}
// 主线程等待2s,等待测试线程获得锁后执行
TimeUnit.SECONDS.sleep(2);
print(thread,"测试线程最终状态:");
}
}