前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java线程的那些状态

Java线程的那些状态

作者头像
luoxn28
发布2021-04-08 14:28:17
3300
发布2021-04-08 14:28:17
举报
文章被收录于专栏:TopCoderTopCoder

Java线程(或者说计算机线程),是有一些状态来表示当前线程运行信息的,可以通过jsatck命令来查看Java进程中线程函数栈信息,其中就包括了Java线程状态。

在分析Java线程状态之前,我们先来看下进程、线程、协程这些概念之间的区别:

  • 进程:运行时程序的抽象,系统资源管理的基本单位;
  • 线程:一个进程可以包含多个线程,CPU调度执行的基本单位,Linux下称线程为轻量级进程;
  • 协程:一种比线程更加轻量级的存在,线程是由操作系统内核负责管理调度的,而协程一般是由用户进程负责调度管理,多个协程之间可使用同一个线程来运行,比如go协程的实现就是这样的。

对于开发小伙伴来讲,了解Java线程状态,有利于加深对线程的理解,有助于解决线程死锁、线程阻塞等问题。

Java Thread类型的State枚举就定义了如下6种线程状态,这些状态之间会进行切换直到线程终止为止,类似于状态机流转。

代码语言:javascript
复制
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;
}

这些Java线程状态图如下所示:

注意,RUNNABLE状态时的线程不一定就是RUNNING的,有可能CPU还在执行其他线程而没有调度到该线程。

我们知道了Java层面对于线程状态的几种定义,那么Java/JDK底层是基于什么机制来实现线程管理的呢?下面咱们写个示例Demo进行分析:

代码语言:javascript
复制
@SuppressWarnings("all")
public class MainDemo {

    private static Object obj1 = new Object();
    private static Lock lock = new ReentrantLock();

    public static void main(String[] args) throws InterruptedException {
        new Thread(new Runnable() {
            @Override
            public void run() {
                LockSupport.park();
            }
        }, "t1").start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                LockSupport.park(TimeUnit.DAYS.toNanos(1));
            }
        }, "t2").start();

        lock.lock();
        new Thread(new Runnable() {
            @Override
            public void run() {
                lock.lock();
                System.out.println("zzz222");
            }
        }, "t3").start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(1000000000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "t4").start();

        synchronized (obj1) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    synchronized (obj1) {
                        System.out.println("zzz");
                    }
                }
            }, "t5").start();

            Thread.sleep(1000000000);
        }
    }
}

上述代码起了5个子线程,名字分别是t1、t2到t5,其中t1执行LockSupport.park()阻塞,t2执行LockSupport.park(long)阻塞,t3执行lock.lock()阻塞,t4执行Thread.sleep()阻塞,t5执行synchronized(obj)阻塞。

通过jstack命令可以查看Java线程栈,如下所示:

但是它不能看到JDK和系统层面的线程栈,此时可以通过pstack命令来查看,如下所示:

那么Java线程栈怎么和系统层线程栈关联起来呢?这个时候可以通过Java线程栈中nid(十六进制)和系统层线程栈中LWP是否相等关联起来。比如t1线程的线程栈如下所示:

通过对t1-t5线程栈进行分析,这些线程都处于TIMED_WAITING或者WAITING状态,它们的JDK层面都是走到了方法pthread_cond_waitpthread_cond_timedwait,而这些就是系统层pthread库中方法了,对于学过 Linux C 的同学肯定都不会陌生。感兴趣的同学可以按照上面的命令查看下所有线程的整个线程栈信息。

推荐阅读

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2021-03-20,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 TopCoder 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档