一、线程状态的枚举

Java的线程从创建到销毁总共有6种状态。这些状态被定义在Thread类种的内部枚举 State 中,分别如下:

1、NEW:初始状态。

线程实例已创建,但未启动。

// 实例创建后,初始状态为 NEW
Thread thread = new Thread();

2、RUNNABLE:运行状态。

线程正在JVM中执行或等待操作系统资源(如CPU),包含 Ready(就绪)状态和 Running(运行中)状态。

(1)Ready 状态:该线程在可运行的状态,但在此刻线程调度器并没有选择执行该线程。

(2)Running 状态:此刻线程调度器选择执行该线程,线程得到了CPU的时间片资源。

3、BLOCKED:阻塞状态。

线程在等待获取监视器锁资源,以便进入 synchronized 标记的方法或者代码块。

4、WAITING:等待状态。

当调用以下方法后,线程将进入等待状态:

(1)Object.wait(); // 调用不带超时参数的 wait() 方法。

(2)anotherThread.join(); // 调用另一个线程的不带超时参数的 join() 方法。

(3)LockSupport.park(); // 无限期挂起当前线程。

5、TIMED_WAITING:超时等待状态。

指定了超时时间的等待状态。当线程调用了如下方法后,线程将进入超时等待状态:

(1)Thread.sleep(long millis)Thread.sleep(long millis, int nanos)

(2)Object.wait(long timeout)Object.wait(long timeout, int nanos)

(3)anotherThread.join(long millis) anotherThread.join(long millis, int nanos)

(4)LockSupport.parkNanos(Object blocker, long nanos)LockSupport.parkUntil(Object blocker, long deadline)

可以看到,跟 WAITING 状态相比,引起 TIMED_WAITING 状态的方法,主要是多了超时参数。

6、TERMINATED:终止状态。

线程执行完成或被中断,将进入终止状态。进入终止状态后的线程,无法重启启动。

如果再次调用 start() 方法,将会抛出 IllegalThreadStateException 异常。

调用run() 方法也不会有任何效果,线程内部的 Runnable 对象也已经被释放。

二、线程状态的转换

Java线程状态的转换过程,如下图所示。

三、线程状态的获取

当前线程的状态可以通过 Thread.currentThread().getState() 获取。显然,获取自己的状态,得到的肯定是运行状态。

一个线程实例的状态,可以通过 thread.getState()方法获取。

public class Main {
public static void main(String[] args) throws InterruptedException {
Thread thread = new Thread(() -> {
// running:RUNNABLE
System.out.println("running:" + Thread.currentThread().getState());
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
});
// before start():NEW
System.out.println("before start():" + thread.getState());
thread.start();
// 暂停一下主线程
Thread.sleep(100);
// 由于线程内sleep了1秒,此时将处于有限时间等待状态
// after start():TIMED_WAITING
System.out.println("after start():" + thread.getState());
// 等待线程结束
thread.join();
// thread state:TERMINATED
System.out.println("thread state:" + thread.getState());
}
}

测试Lock对应的线程状态:

public class Main {
private static final Lock LOCK = new ReentrantLock();
private static volatile int value = 0;

public static void main(String[] args) throws InterruptedException {
Thread alice = new Thread(() -> testLock("Alice"));
Thread bob = new Thread(() -> testLock("Bob"));
// alice before start():NEW
System.out.println("alice before start():" + alice.getState());
alice.start();
// 暂停一下主线程
Thread.sleep(100);
bob.start();
Thread.sleep(100);
// bob.getState():WAITING
System.out.println("bob.getState():" + bob.getState());
value = 1000;
}

static void testLock(String name) {
// 注意:锁住的是代码块,不是里面用到的变量、资源
LOCK.lock();
try {
System.out.println("Lock block code, name=" + name);
// 注意:这是非线程安全的操作
value += 10;
System.out.println("Before sleep(): name=" + name + ",value=" + value);
Thread.sleep(1000);
System.out.println("After sleep(): name=" + name + ",value=" + value);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
LOCK.unlock();
}
}
}

由上面的例子可见,等待进入Lock.lock() 代码块的线程的状态,是 WAITING 状态,而不是 BLOCKED。

如果调用的是Lock.tryLock(long, TimeUnit) 方法,对应的状态将是 TIMED_WAITING。

【Java】线程的 6 种状态的更多相关文章

  1. java 线程的几种状态(转载)

      java thread的运行周期中, 有几种状态, 在 java.lang.Thread.State 中有详细定义和说明: NEW 状态是指线程刚创建, 尚未启动 RUNNABLE 状态是线程正在 ...

  2. JAVA 线程的6种状态

    JAVA线程的6种状态 线程状态(Thread.State).线程处于下列状态的其中之一. 一个线程在一个时刻只能有一个状态.这些状态是虚拟机线程状态,不能反应任何操作系统的线程状态. 通过Threa ...

  3. 并发编程——Java线程的6种状态及切换

    前言 本次主要分享一下Java线程的六种状态及其转换. 如果对于线程的创建方式不太了解,推荐观看并发编程--认识java里的线程 线程的状态及其转换 操作系统线程的五种状态 新建(NEW) 就绪(RU ...

  4. java 线程的几种状态

    java thread的运行周期中, 有几种状态, 在 java.lang.Thread.State 中有详细定义和说明: NEW 状态是指线程刚创建, 尚未启动 RUNNABLE 状态是线程正在正常 ...

  5. 透彻讲解,Java线程的6种状态及切换

    Java中线程的状态分为6种. 1. 初始(NEW):新创建了一个线程对象,但还没有调用start()方法.2. 运行(RUNNABLE):Java线程中将就绪(ready)和运行中(running) ...

  6. Java线程的5种状态及切换(透彻讲解)-京东面试

    一.Thread的几个重要方法: 我们先了解一下Thread的几个重要方法. a.start()方法,开始执行该线程:b.stop()方法,强制结束该线程执行:c.join方法,等待该线程结束.d.s ...

  7. 【转】java 线程的几种状态

    java thread的运行周期中, 有几种状态, 在 java.lang.Thread.State 中有详细定义和说明: NEW 状态是指线程刚创建, 尚未启动 RUNNABLE 状态是线程正在正常 ...

  8. Java线程的5种状态及切换

    ava中的线程的生命周期大体可分为5种状态. 1. 新建(NEW):新创建了一个线程对象. 2. 可运行(RUNNABLE):线程对象创建后,其他线程(比如main线程)调用了该对象的start()方 ...

  9. Java线程的6种状态及切换

    Java中线程的状态分为6种. 1. 初始(NEW):新创建了一个线程对象,但还没有调用start()方法.2. 运行(RUNNABLE):Java线程中将就绪(ready)和运行中(running) ...

  10. Java线程的6种状态

    6种状态分别是: NEW.RUNNABLE.TERMINATED.WAITING.TIMED_WAITING.BLOCKED NEW:线程创建完毕 RUNNABLE:线程运行中,又分为READY + ...

随机推荐

  1. [学习笔记] 数位DP的dfs写法

    跟着洛谷日报走,算法习题全都有! 嗯,没错,这次我也是看了洛谷日报的第84期才学会这种算法的,也感谢Mathison大佬,素不相识,却写了一长篇文章来帮助我学习这个算法. 算法思路: 感觉dfs版的数 ...

  2. 多台centos7服务器实现SSH免密登陆

    一.环境 centos7.x 三台 node1.node2.node3 二.实现免密登陆 2.1.node1上,生成公钥与私钥 [root@node1 ~]# ssh-keygen Generatin ...

  3. 线程基本使用--Thread内部方法调用start

    一个问题,下面的代码会如何运行 public class TraditionalThread { public static void main(String[] args) { System.out ...

  4. 手把手教你AspNetCore WebApi:缓存(MemoryCache和Redis)

    前言 这几天小明又有烦恼了,系统上线一段时间后,系统性能出现了问题,马老板很生气,叫小明一定要解决这个问题.性能问题一般用什么来解决呢?小明第一时间想到了缓存. 什么是缓存 缓存是实际工作中非常常用的 ...

  5. Linux Centos7 安装Docker-CE

    先确保yum 是最新版本 执行: sudo yum update 添加docker源地址 sudo yum-config-manager --add-repo https://download.doc ...

  6. C++字符串操作小结

    忽略大小写比较大小 库函数strcasecmp和_stricmp: 这两个函数都不属于C++标准库,strcasecmp由POSIX引入,windows平台则定义了功能等价的_stricmp.用法和C ...

  7. 习题3-5 谜题(Puzzle, ACM/ICPC World Finals 1993, UVa227)

    #include<stdio.h> #include<string.h> char s[5][5]; int main() { while(gets(s[0])) { int ...

  8. kafka-伪集群搭建

      一.简介 Apache Kafka是一个快速.可扩展的.高吞吐的.可容错的分布式"发布-订阅"消息系统,使用Scala与Java语言编写,能够将消息从一个端点传递到另一个端点, ...

  9. spring boot: 设计接口站api的版本号,支持次版本号(spring boot 2.3.2)

    一,为什么接口站的api要使用版本号? 1,当服务端接口的功能发生改进后, 客户端如果不更新版本,    则服务端返回的功能可能不能使用,    所以在服务端功能升级后,     客户端也要相应的使用 ...

  10. centos8平台给sudo配置日志

    一,sudo日志的用途: 我们可以记录下来用户账号在哪个时间进行过sudo 这样不需要再从secure日志中查找用户的sudo记录 说明:刘宏缔的架构森林是一个专注架构的博客,地址:https://w ...