当前位置: 首页 > news >正文

线程状态及转换详解

在Java中,线程的状态由 java.lang.Thread.State 枚举定义

每个线程在其生命周期内会经历不同的状态,并且这些状态之间可以相互转换


1. 线程的六种状态

根据 Thread.State 枚举,Java线程有以下六种状态:

(1) NEW(新建)

  • 描述:线程对象已经被创建,但尚未调用 start() 方法启动

  • 特点:

    • 线程尚未开始运行
    • 此时线程处于初始状态
  • 示例:

    Thread thread = new Thread(() -> System.out.println("Hello"));
    // 当前线程状态为 NEW
    

(2) RUNNABLE(可运行)

  • 描述:线程已经启动,正在 JVM 中执行,或者正在等待操作系统的资源(如CPU时间片)

  • 特点:

    • 线程可能正在运行,也可能在等待CPU调度
    • 在操作系统层面,线程可能处于运行状态或就绪状态
  • 转换条件:

    • 调用 start() 方法后,线程从 NEW 状态进入 RUNNABLE 状态
  • 示例:

    Thread thread = new Thread(() -> System.out.println("Running"));
    thread.start();
    // 当前线程状态为 RUNNABLE
    

(3) BLOCKED(阻塞)

  • 描述:线程在尝试获取一个被其他线程持有的锁时进入此状态

  • 特点:

    • 线程正在等待进入同步代码块或方法
    • 一旦锁可用,线程将重新进入 RUNNABLE 状态
  • 转换条件:

    • 线程试图获取一个已被其他线程占用的锁
  • 示例:

    synchronized (lock) {
        // 如果另一个线程持有 lock,则当前线程进入 BLOCKED 状态
    }
    

(4) WAITING(等待)

  • 描述:线程在等待另一个线程显式地唤醒它

  • 特点:

    • 线程无限期等待,直到被其他线程唤醒
    • 常见的触发方法包括:
      • Object.wait()(无超时版本)
      • Thread.join()(无超时版本)
      • LockSupport.park()
  • 转换条件:

    • 调用上述方法后,线程从 RUNNABLE 状态进入 WAITING 状态
    • 被其他线程唤醒后,线程重新进入 RUNNABLE 状态
  • 示例:

    Object lock = new Object();
    synchronized (lock) {
        lock.wait(); // 当前线程进入 WAITING 状态
    }
    

(5) TIMED_WAITING(计时等待)

  • 描述:线程在等待一段时间后自动恢复,或者被其他线程显式唤醒

  • 特点:

    • 线程等待的时间是有限的。
    • 常见的触发方法包括:
      • Thread.sleep(long millis)
      • Object.wait(long timeout)
      • Thread.join(long millis)
      • LockSupport.parkNanos()
      • LockSupport.parkUntil()
  • 转换条件:

    • 调用上述方法后,线程从 RUNNABLE 状态进入 TIMED_WAITING 状态。
    • 等待时间结束后,线程重新进入 RUNNABLE 状态。
  • 示例:

    Thread.sleep(1000); // 当前线程进入 TIMED_WAITING 状态,持续1秒
    
(6) TERMINATED(终止)
  • 描述:线程已经完成执行,或者因异常而终止

  • 特点:

    • 线程的生命周期结束
    • 此时线程对象仍然存在,但无法再次启动
  • 转换条件:

    • 线程的 run() 方法执行完毕
    • 线程因未捕获的异常而终止
  • 示例:

    Thread thread = new Thread(() -> System.out.println("Done"));
    thread.start();
    thread.join(); // 等待线程结束
    // 当前线程状态为 TERMINATED
    

2. 线程状态的转换图

以下是 Java 线程状态之间的转换关系:

       start()
NEW --------------> RUNNABLE <-----------------------+
        |                         ^                   |
        |                         |                   |
        |                      sleep()                |
        |                         |                   |
        |                         v                   |
        |                 TIMED_WAITING               |
        |                         |                   |
        |                         +-------------------+
        |                         |
        |                    wait()/join()
        |                         |
        |                         v
        |                     WAITING
        |                         |
        |                         +-------------------+
        |                         |                   |
        |         获取锁           |                   |
        |                         v                   |
        | BLOCKED -------------> RUNNABLE -----------> TERMINATED
                                  ^                   ^
                                  |                   |
                             run()结束/异常          |

3. 状态转换详解

(1) NEW → RUNNABLE

  • 调用 start() 方法后,线程从 NEW 状态进入 RUNNABLE 状态
  • 注意:直接调用 run() 方法不会启动新线程,而是作为普通方法在当前线程中执行

(2) RUNNABLE ↔ BLOCKED

  • 当线程尝试进入同步代码块或方法,但锁被其他线程占用时,线程进入 BLOCKED 状态
  • 一旦锁可用,线程重新进入 RUNNABLE 状态

(3) RUNNABLE ↔ WAITING

  • 调用 wait()join()park() 方法后,线程进入 WAITING 状态。
  • 被其他线程显式唤醒(如调用 notify()unpark())后,线程重新进入 RUNNABLE 状态。

(4) RUNNABLE ↔ TIMED_WAITING

  • 调用 sleep()wait(long timeout)join(long millis) 方法后,线程进入 TIMED_WAITING 状态
  • 等待时间结束后,线程重新进入 RUNNABLE 状态

(5) RUNNABLE → TERMINATED

  • 线程的 run() 方法执行完毕,或者因未捕获的异常而终止,线程进入 TERMINATED 状态

4. 总结

状态描述典型方法
NEW线程对象已创建,但尚未启动new Thread()
RUNNABLE线程正在运行或等待CPU调度start()
BLOCKED线程等待获取锁synchronized
WAITING线程无限期等待另一个线程唤醒wait()join()park()
TIMED_WAITING线程等待一段时间后自动恢复或被唤醒sleep()wait(long)join(long)
TERMINATED线程执行完毕或因异常终止run() 结束

文章转载自:
http://cerograph.bdypl.cn
http://agreeably.bdypl.cn
http://amenorrhoea.bdypl.cn
http://bacchanalian.bdypl.cn
http://ball.bdypl.cn
http://bemock.bdypl.cn
http://calender.bdypl.cn
http://athonite.bdypl.cn
http://absquatulation.bdypl.cn
http://barometrical.bdypl.cn
http://chameleon.bdypl.cn
http://ascender.bdypl.cn
http://chaffcutter.bdypl.cn
http://casing.bdypl.cn
http://brahmanist.bdypl.cn
http://astrictive.bdypl.cn
http://bfc.bdypl.cn
http://allahabad.bdypl.cn
http://blister.bdypl.cn
http://cainite.bdypl.cn
http://abduce.bdypl.cn
http://ankylostomiasis.bdypl.cn
http://alchemistic.bdypl.cn
http://acuminous.bdypl.cn
http://biquarterly.bdypl.cn
http://aram.bdypl.cn
http://chieftain.bdypl.cn
http://biflex.bdypl.cn
http://baleful.bdypl.cn
http://bounder.bdypl.cn
http://www.dtcms.com/a/94369.html

相关文章:

  • 【大模型基础_毛玉仁】5.1 模型编辑简介
  • DCAT模型:双交叉注意力革新医学影像诊断,AUC 99.75%
  • spring security整体架构
  • 【Python语言基础】13、函数-03
  • day19
  • 【无标题】Java的基础准备
  • 基于python爬虫:requests+BeautifulSoup+MySQL/MongoDB(或:CSV、JSON等格式的文件)+...
  • Java版Manus实现来了,Spring AI Alibaba发布开源OpenManus实现
  • MySQL执行计划
  • 固定资产管理如何适配不同规模企业的发展需求?
  • C++中的智能指针
  • 多 线 程
  • 单应性矩阵(homography)
  • deepseek实战教程--第七篇DS大模型与AI平台千丝万缕的关系
  • 前端开发中生成网站的favicon.ico文件的软件推荐及使用方法
  • 有效三角形的个数
  • Redis的基础,经典,高级问题解答篇
  • DeepSeek 助力 Vue3 开发:打造丝滑的表格(Table)之添加行拖拽排序功能示例6,TableView16_06 分页表格拖拽排序
  • 【洛谷题单】暴力枚举(上)
  • 淳厚的sql注入内功
  • [特殊字符] Hyperlane:Rust 高性能 Web 框架的终极选择 [特殊字符]
  • 新能源动力电池测试设备深度解析:充放电设备与电池模拟器的差异及技术趋势
  • C#从入门到精通(3)
  • 栈-有效的括号
  • 计算机网络八股
  • 第五周日志-重新学汇编(2)
  • dify+vue+java接入大模型流式输出
  • 面试计算机操作系统解析(一中)
  • MySQL 优化利器 SHOW PROFILE 的实现原理
  • IP组播 C++简单应用