Java线程状态如何流转?
时间:2026-05-20 17:46:21 399浏览 收藏
Java线程的六种状态(NEW、RUNNABLE、BLOCKED、WAITING、TIMED_WAITING、TERMINATED)并非教科书式的理想化流转,而是深度耦合JVM抽象与操作系统调度的动态过程:NEW仅瞬时存在于new之后start之前,不可中断且难以观测;RUNNABLE实为“可运行态”而非“正在运行”,涵盖就绪、执行甚至部分I/O等待,极易被误读为CPU高负载;WAITING与TIMED_WAITING的本质区别在于是否依赖外部唤醒或超时自动恢复,常见于并发工具底层;而BLOCKED严格限定于synchronized锁竞争,与Lock机制无关——理解这些微妙差异,远比死记状态图更能帮你精准定位线程挂起、假死或资源争用等真实生产问题。

NEW 状态只在 new Thread() 之后、start() 之前存在
刚用 new Thread(runnable) 创建出来的线程对象,还没调用 start(),它就处于 NEW 状态。这个状态非常短暂,且不可被调度执行。一旦调用 start(),JVM 就会为其分配资源并尝试进入就绪队列——此时状态立刻变为 RUNNABLE(注意不是 “RUNNING”)。
NEW线程无法通过Thread.getState()被外部观察到正在运行,甚至不能被中断(interrupt()无效)- 重复调用
start()会抛出IllegalThreadStateException,这是唯一能明确感知该线程已离开NEW的信号 - 不要依赖
getState() == NEW来判断线程是否“还没开始”,因为从 new 到 start 的间隙极短,且无内存屏障保证可见性
RUNNABLE 并不等于“正在 CPU 上执行”
RUNNABLE 是 Java 线程状态中覆盖范围最广的一个:它包含操作系统层面的 “ready”(就绪)、“running”(运行中),也包括 I/O 阻塞但未进入 WAITING 或 BLOCKED 的情况(比如调用 FileInputStream.read() 时内核态等待磁盘数据,JVM 仍认为它是 RUNNABLE)。
- 线程池中的空闲工作线程,只要没被
park或wait,就始终是RUNNABLE(在ThreadPoolExecutor的runWorker循环里阻塞在getTask(),但该方法内部用的是LockSupport.parkNanos,所以实际状态是WAITING—— 这是个常见误解点) Thread.getState()返回RUNNABLE时,不能推断出 CPU 使用率高;同样,CPU 占用低也不代表状态不是RUNNABLE- Java 10+ 引入了
Thread.getState()的 JVM TI 支持增强,但底层仍是 OS 调度视角和 JVM 抽象层的混合映射,别把它当精确的运行时快照用
WAITING 和 TIMED_WAITING 的触发方式完全不同
WAITING 是无限期等待,必须靠其他线程显式唤醒;TIMED_WAITING 是带超时的等待,时间一到自动返回。两者都常见于同步工具类内部,但触发路径差异明显:
WAITING:由Object.wait()(无参)、Thread.join()(无参)、LockSupport.park()触发TIMED_WAITING:由Object.wait(timeout)、Thread.sleep(millis)、Thread.join(timeout)、LockSupport.parkNanos(ns)、Condition.awaitNanos()触发- 特别注意:
Thread.sleep(0)是合法的,它会让出当前时间片,状态为TIMED_WAITING,但不会真正休眠 —— 这在自旋退避逻辑里有时被误用 - JVM 线程 dump 中看到大量
TIMED_WAITING (parking),大概率是java.util.concurrent类(如LinkedBlockingQueue)在用LockSupport.parkNanos实现超时等待,属正常行为
BLOCKED 只发生在进入 synchronized 同步块/方法时竞争 monitor 锁
BLOCKED 是唯一与 synchronized 关键字强绑定的状态。它表示线程已准备好运行,但正等待获取某个对象的 monitor 锁(即进入 synchronized(obj) { ... } 或调用 synchronized 方法时,发现锁被别的线程持有)。
ReentrantLock.lock()不会导致BLOCKED,而是让线程进入WAITING或TIMED_WAITING(取决于是否用tryLock(long, TimeUnit))- 一个线程可以在持有某对象锁的同时,因等待另一个对象锁而变成
BLOCKED,这就是死锁链的起点 - 线程 dump 中若出现多个线程相互
BLOCKED on,基本可判定发生锁竞争或死锁;但要注意,BLOCKED时间极短(纳秒级)时,dump 可能抓不到,需结合jstack -l和Unsafe.park堆栈综合判断
RUNNABLE ↔ TIMED_WAITING 切换频繁,BLOCKED 可能一闪而过,而 WAITING 一旦卡住往往意味着业务逻辑或资源泄漏。看状态不如看堆栈,查问题先看 jstack 输出里最顶上的 native 方法和锁地址。好了,本文到此结束,带大家了解了《Java线程状态如何流转?》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多文章知识!
相关阅读
更多>
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
最新阅读
更多>
-
186 收藏
-
189 收藏
-
108 收藏
-
343 收藏
-
182 收藏
-
492 收藏
-
321 收藏
-
442 收藏
-
174 收藏
-
304 收藏
-
350 收藏
-
399 收藏
课程推荐
更多>
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习