再来捋捋Java 线程状态

前言

线程并发系列文章:

Java 线程基础
Java 线程状态
Java “优雅”地中断线程-实践篇
Java “优雅”地中断线程-原理篇
真正理解Java Volatile的妙用
Java ThreadLocal你之前了解的可能有误
Java Unsafe/CAS/LockSupport 应用与原理
Java 并发"锁"的本质(一步步实现锁)
Java Synchronized实现互斥之应用与源码初探
Java 对象头分析与使用(Synchronized相关)
Java Synchronized 偏向锁/轻量级锁/重量级锁的演变过程
Java Synchronized 重量级锁原理深入剖析上(互斥篇)
Java Synchronized 重量级锁原理深入剖析下(同步篇)
Java并发之 AQS 深入解析(上)
Java并发之 AQS 深入解析(下)
Java Thread.sleep/Thread.join/Thread.yield/Object.wait/Condition.await 详解
Java 并发之 ReentrantLock 深入分析(与Synchronized区别)
Java 并发之 ReentrantReadWriteLock 深入分析
Java Semaphore/CountDownLatch/CyclicBarrier 深入解析(原理篇)
Java Semaphore/CountDownLatch/CyclicBarrier 深入解析(应用篇)
最详细的图文解析Java各种锁(终极篇)
线程池必懂系列

在现代操作系统里,有进程和线程的概念,我们先来简单了解一下这两个概念。

  • 进程是资源分派的基本单位
  • 线程是cpu调度基本单位

刚开始,计算机系统比较简单,设计者把cpu上执行的每个任务抽象为进程,操作系统统一管理进程的调度,给每个进程分配地址空间、外设等,这些统称为进程的资源。当cpu执行A进程过程中,发现A的时间片已经耗尽,因此先将A进程挂起(释放cpu),并保存进程A的上下文,然后再调度进程B。后来发现每次切换进程的上下文都比较耗时,浪费有限的系统资源,因此将进程再划分为粒度更小的单位,称之为线程(实际上是cpu上的一个执行流),进程内的线程共享进程的资源,因此cpu切换线程所花代价更小。

Linux进程状态

cpu同一时刻只能处理一个进程,有可能进程还在排队等待cpu执行,可能在某个地方等待资源(如输入事件等),因此操作系统给予进程状态来标识进程某个时刻的状态。[1]


image.png

如上图,进程状态为ready(就绪),running(运行中),waiting(等待)
实际上线程状态也可以类比进程状态。

Java线程状态

Java是门支持多线程的语言,Java线程是JVM实现的,JVM是OS的一个进程,Java线程对于OS来说是透明的。接下来我们就Java线程类Thread入手了解一下Java线程状态。

    public enum State {
        /**
         * Thread state for a thread which has not yet started.
         */
        NEW,

        /**
         * Thread state for a runnable thread.  A thread in the runnable
         * state is executing in the Java virtual machine but it may
         * be waiting for other resources from the operating system
         * such as processor.
         */
        RUNNABLE,

        /**
         * Thread state for a thread blocked waiting for a monitor lock.
         * A thread in the blocked state is waiting for a monitor lock
         * to enter a synchronized block/method or
         * reenter a synchronized block/method after calling
         * {@link Object#wait() Object.wait}.
         */
        BLOCKED,

        /**
         * Thread state for a waiting thread.
         * A thread is in the waiting state due to calling one of the
         * following methods:
         * <ul>
         *   <li>{@link Object#wait() Object.wait} with no timeout</li>
         *   <li>{@link #join() Thread.join} with no timeout</li>
         *   <li>{@link LockSupport#park() LockSupport.park}</li>
         * </ul>
         *
         * <p>A thread in the waiting state is waiting for another thread to
         * perform a particular action.
         *
         * For example, a thread that has called <tt>Object.wait()</tt>
         * on an object is waiting for another thread to call
         * <tt>Object.notify()</tt> or <tt>Object.notifyAll()</tt> on
         * that object. A thread that has called <tt>Thread.join()</tt>
         * is waiting for a specified thread to terminate.
         */
        WAITING,

        /**
         * Thread state for a waiting thread with a specified waiting time.
         * A thread is in the timed waiting state due to calling one of
         * the following methods with a specified positive waiting time:
         * <ul>
         *   <li>{@link #sleep Thread.sleep}</li>
         *   <li>{@link Object#wait(long) Object.wait} with timeout</li>
         *   <li>{@link #join(long) Thread.join} with timeout</li>
         *   <li>{@link LockSupport#parkNanos LockSupport.parkNanos}</li>
         *   <li>{@link LockSupport#parkUntil LockSupport.parkUntil}</li>
         * </ul>
         */
        TIMED_WAITING,

        /**
         * Thread state for a terminated thread.
         * The thread has completed execution.
         */
        TERMINATED;
    }

注释比较清晰,这6种状态涵盖了Thread整个生命周期,简单概括一下:

1、NEW:构造了thread实例,但是还没有start
2、RUNNABLE:线程正在运行或者正等待被cpu执行
3、BLOCKED:线程调用synchronized关键字等待获取monitor锁
4、WAITING:线程调用了无超时的wait、join、park方法
5、TIMED_WAITING:线程调用了有超时的wait、sleep、join、parkNanos、parkUntil方法
6、TERMINATED:线程终止/完成了运行

线程各种状态之间是如何转换的呢?用图说明:[2]


image.png

这张图清晰展示了各个状态之间的关系,大家可能注意到图上多了两个状态:Ready和Running,而少了RUNNABLE状态。还记得之前说的RUNNABLE:“线程正在运行或者正等待被cpu执行”,实际上RUNNABLE可以认为是Ready和Running状态的结合体,Ready表示就绪状态,Running表示正在运行的状态。

代码验证状态

上面的状态描述可能比较抽象,Java 1.5之后有个方法:getState() 用来获取线程的状态,因此我们可以打印该值来观察状态之间的流转。

验证BLOCKED状态

public class TestThreadState {

    public static void main(String args[]) {
        final Object object = new Object();
        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                synchronized (object) {
                    System.out.println("t1 getted lock");
                    while(true);
                }
            }
        }, "t1");

        Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                synchronized (object) {
                    System.out.println("t2 getted lock");
                }
            }
        }, "t2");

        //test state
        t1.start();
        sleep(2);
        t2.start();

        while(true) {
            System.out.println("t1 state:" + t1.getState());
            System.out.println("t2 state:" + t2.getState());
            sleep(2);
        }

    }

    private static void sleep(long time) {
        try {
            TimeUnit.SECONDS.sleep(time);
        } catch (Exception e) {
        }
    }

}
image.png

t1先获得monitor lock,并一直循环不释放锁,此时状态为RUNNABLE;t2尝试获取monitor lock失败,并阻塞,此时状态为BLOCKED。
需要注意的是,如果使用ReentrantLock,线程因ReentrantLock阻塞,此时线程状态为WAITING。

验证WAITING/TIMED_WAITING状态

将上面的例子稍微修改为:

        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                synchronized (object) {
                    System.out.println("t1 getted lock");
                    try {
                        object.wait();
                        System.out.println("t1 get signal from t2");
                    } catch (Exception e) {

                    }
                }
            }
        }, "t1");

        Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                synchronized (object) {
                    System.out.println("t2 getted lock");
                    sleep(3);
                    System.out.println("t2 notify");
                    try {
                        object.notify();
                    } catch (Exception e) {

                    }
                    //先睡眠
                    sleep(3);
                }
            }
        }, "t2");
image.png

t1先获得锁,并调用wait阻塞线程和放弃锁,此时状态为:WAITING。t2获取锁成功,并睡眠3s,此时状态为:TIMED_WAITING。t2睡眠结束后调用notify唤醒t1,此时t2继续睡眠,状态为:TIMED_WAITING。t1收到notify信息后,尝试获取锁,但由于此时t2还在睡眠没有释放锁,因此t1被阻塞,状态为BlOCKED。t2睡眠结束,释放锁,t1获得锁并接着打印“t1 get signal from t2”语句,最终t1、t2终止了运行,处在TERMINATED状态。

Join和Yield

Join是Thread类成员方法,该方法作用是当前线程等待某个线程结束后再继续执行后续代码。
Yield是Thread类静态方法,该方法作用是让当前线程放弃cpu,并等待cpu重新调度(当前线程可能再次获得cpu)。通俗点的说法是给其它线程一个机会,让大家回到同一起跑线,至于谁被cpu选中,看个人造化。

public class TestJoin {
    public static void main(String args[]) {
        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(3000);
                    System.out.println("t1 end");
                } catch (Exception e) {

                }
            }
        }, "t1");
        t1.start();

        try {
            System.out.println("wait t1 terminate");
            t1.join();
        } catch (Exception e) {

        }

        System.out.println("main thread end");
    }
}

main thread等待t1执行完成再继续执行。
网上不少文章列举Yield用法示例:

public class TestYield {
    public static void main(String args[]) {
        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 5; i++) {
                    if (i == 3) {
                        Thread.yield();
                    }
                    System.out.println("t1->" + i);
                }
            }
        }, "t1");
        t1.start();

        Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 5; i++) {
                    System.out.println("t2->" + i);
                }
            }
        }, "t2");
        t2.start();
    }
}

t1在执行期间调用yield()方法,放弃cpu,这时候t2就可以拿到cpu。实际上打印结果的顺序充满不确定性,不能完全确定是由于yield()方法引起的差异,因此上面的例子不够充分。而我们平时使用yield()方法的情况很少,基本无需关注。

LockSupport

用于线程阻塞与唤醒,更详细的解释请查看:Java Unsafe/CAS/LockSupport 应用与原理

线程状态的意义

在开发的过程中,免不了用到多线程。多线程之间的协作(同步、互斥)需要正确有序的进行,在程序运行过程中如果出现了多线程问题一般都比较随机不容易复现(死锁等),这个时候可以通过打印线程状态(jstack)来分析各个线程状态,到底处在什么状态?因为什么原因处在这个状态?实际应该进行到哪个状态?这样对于问题的分析就会做到有的放矢,提高效率。
对如何中断线程感兴趣的同学请移步:Java “优雅”地中断线程

参考文章

[1] https://www.tecmint.com/linux-process-management/
[2] https://www.uml-diagrams.org/examples/java-6-thread-state-machine-diagram-example.html

您若喜欢,请点赞、关注,您的鼓励是我前进的动力

持续更新中,和我一起步步为营系统、深入学习Android

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容