Java线程生命周期的终止与复位

目录
  • Thread生命周期
    • 生命周期概述
    • 线程生命周期流程图
    • 线程生命周期测试
  • 启动线程
    • java中的启动
    • Hotspot中的启动
  • 线程中断与复位
    • 不要使用stop方法
    • 使用interrupt方法
    • 线程的复位
    • 其他的线程复位

Thread生命周期

生命周期概述

Java的线程状态描述放在Thread类里面的枚举类State中.总共包含了6中状态(从出生到死亡)。

public enum State {
    /**
     * 尚未启动的线程的线程状态 (没有start)
     */
    NEW,

    /**
     * 可运行线程的线程状态,是可以运行的线程状态(并不是在运行)
     * 这个状态在Java虚拟机中进行,但它可能等待来自操作系统的其他资源,比如CPU。
     * 内部包含了两个状态 【RUNNING】,【READY】这两个状态是可以互相流转的
     * 调用了start后线程就处于 READY 状态 ,等待操作系统分配CPU时间片,分配后进入 RUNNING 状态。
     * 当调用 yield() 方法后,只是谦让的允许当前线程让出 CPU ,但是不一定让,由操作系统决定,如果让      * 了当前线程就会进入 READY 状态,等待系统分配CPU时间片再次进入 RUNNING 状态。
     */
    RUNNABLE,

    /**
     * 阻塞状态。
     * 线程阻塞,等待监视器锁的状态,获取监视器锁后会进入 RUNNABLE  状态
     * 当发生线程锁竞争状态下,没有获取到锁的线程会被挂起进入阻塞状态,比如synchronized锁。
     */
    BLOCKED,

    /**
     * 等待线程的线程状态
     * 线程调用以下方法会处于等待状态:Object.wait()不超时、Thread.join()不超时等方法
     * 一个处于等待状态的线程正在等待另一个线程执行特定动作,例如:
     * 一个线程调用了Object.wait()方法在一个对象上正在等待另一个线程调用Object.nofify()或者
     * Object.nofifyAll()方法开启那个对象
     * 一个调用了Thread.join()方法的线程正在等待指定线程终止
     */
    WAITING,

    /**
     * 具有指定等待时间的等待线程的线程状态,调用一下方法会处于这个状态: Object.wait() 超时、          * Thread.join()超时 Thread.sleep(long) 等方法
     */
    TIMED_WAITING,

    /**
     * 已终止线程的线程状态
     * 线程执行完毕或者发生异常终止执行
     */
    TERMINATED;
}

线程生命周期流程图

线程生命周期测试

public class ThreadStatusDemo {
    public static void main(String[] args) throws InterruptedException {
        // 测试 NEW RUNNABLE TERMINATED
        Thread terminated_thread = new Thread(() -> {
            long start = System.currentTimeMillis();
            // 运行三秒 ,打印TERMINATED_THREAD线程runnable状态
            while (System.currentTimeMillis()-start<3000){}
        }, "TERMINATED_THREAD");
        // NEW
        Thread.State state = terminated_thread.getState();
        System.out.println(terminated_thread.getName()+" :state = " + state);

        terminated_thread.start();
        TimeUnit.SECONDS.sleep(1);
        // RUNNABLE
        Thread.State state1 = terminated_thread.getState();
        System.out.println(terminated_thread.getName()+"state1 = " + state1);

        TimeUnit.SECONDS.sleep(5);
        Thread.State state2 = terminated_thread.getState();
        // TERMINATED
        System.out.println(terminated_thread.getName()+"state2 = " + state2);

        // RUNNABLE
        new Thread(() -> {
            while (true) {

            }
        }, "Runnle_Thread").start();
        // TIMED_WAITING
        new Thread(() -> {
            while (true) {
                try {
                    TimeUnit.SECONDS.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "Time_Waiting_Thread").start();
        // WAITING
        new Thread(() -> {
            while (true) {
                synchronized (ThreadStatusDemo.class) {
                    try {
                        ThreadStatusDemo.class.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }, "Waiting_Thread").start();

        // 这两个看谁先抢占到cpu获得锁,另一个就blocked
        // timed_waiting
        new Thread(new BlockedDemo(), "Blocke01_Thread").start();
        // blocked
        new Thread(new BlockedDemo(), "Blocke02_Thread").start();
    }
    static class BlockedDemo extends Thread {
        @Override
        public void run() {
            synchronized (BlockedDemo.class) {
                while (true) {
                    try {
                        TimeUnit.SECONDS.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }
    }
}

启动线程

java中的启动

Java启动一个线程调用start方法,start方法内部调用了 start0()native方法。

public synchronized void start() {
    . . .
    boolean started = false;
    try {
        // 调用native方法
        start0();
        started = true;
    } finally {
        try {
            if (!started) {
                group.threadStartFailed(this);
            }
        } catch (Throwable ignore) {
            /* do nothing. If start0 threw a Throwable then
              it will be passed up the call stack */
        }
    }
}

这个测试是为了验证上图的正确性,只贴了部分.

Hotspot中的启动

查看指引:

jvm.cpp找到JVM_StartThread方法。发现是先创建个 JavaThread作为本地线程然后启动这个本地线程(借助os【thread.cpp】,因为jvm是跨平台的,这里是以linux-os为示例)

JVM_ENTRY(void, JVM_StartThread(JNIEnv* env, jobject jthread))
  JVMWrapper("JVM_StartThread");
  JavaThread *native_thread = NULL;
  bool throw_illegal_thread_state = false;
  {
    MutexLocker mu(Threads_lock);

    if (java_lang_Thread::thread(JNIHandles::resolve_non_null(jthread)) != NULL) {
      throw_illegal_thread_state = true;
    } else {
      jlong size =
             java_lang_Thread::stackSize(JNIHandles::resolve_non_null(jthread));
      size_t sz = size > 0 ? (size_t) size : 0;
      // 先创建一个JavaThread
      native_thread = new JavaThread(&thread_entry, sz);
      if (native_thread->osthread() != NULL) {
        native_thread->prepare(jthread);
      }
    }
  }
  if (throw_illegal_thread_state) {
    THROW(vmSymbols::java_lang_IllegalThreadStateException());
  }
  assert(native_thread != NULL, "Starting null thread?");
  if (native_thread->osthread() == NULL) {
    delete native_thread;
    if (JvmtiExport::should_post_resource_exhausted()) {
      JvmtiExport::post_resource_exhausted(
        JVMTI_RESOURCE_EXHAUSTED_OOM_ERROR | JVMTI_RESOURCE_EXHAUSTED_THREADS,
        "unable to create new native thread");
    }
    THROW_MSG(vmSymbols::java_lang_OutOfMemoryError(),
              "unable to create new native thread");
  }
  // 然后启动这个本地线程 thread.cpp
  Thread::start(native_thread);
JVM_END

JavaThread 创建线程:

JavaThread::JavaThread(ThreadFunction entry_point, size_t stack_sz) :
  Thread()
#if INCLUDE_ALL_GCS
  , _satb_mark_queue(&_satb_mark_queue_set),
  _dirty_card_queue(&_dirty_card_queue_set)
#endif // INCLUDE_ALL_GCS
{
  if (TraceThreadEvents) {
    tty->print_cr("creating thread %p", this);
  }
  initialize();
  _jni_attach_state = _not_attaching_via_jni;
  set_entry_point(entry_point);
  os::ThreadType thr_type = os::java_thread;
  thr_type = entry_point == &compiler_thread_entry ? os::compiler_thread :
                                                     os::java_thread
// 调用os(操作系统)创建个线程
  os::create_thread(this, thr_type, stack_sz);
  _safepoint_visible = false;
   . . .
}

thread.cpp 启动线程:

// tips: 启动线程的方法
void Thread::start(Thread* thread) {
  trace("start", thread);
  // Start is different from resume in that its safety is guaranteed by context or
  // being called from a Java method synchronized on the Thread object.
  if (!DisableStartThread) {
    if (thread->is_Java_thread()) {
      // Initialize the thread state to RUNNABLE before starting this thread.
      // Can not set it after the thread started because we do not know the
      // exact thread state at that time. It could be in MONITOR_WAIT or
      // in SLEEPING or some other state.
      // tips:启动之后设置线程的状态为 可运行状态 RUNNABLE
      java_lang_Thread::set_thread_status(((JavaThread*)thread)->threadObj(),
                                          java_lang_Thread::RUNNABLE);
    }
    // 借助操作系统启动线程
    os::start_thread(thread);
  }
}

线程中断与复位

不要使用stop方法

线程的终止不要简单的调用 stop方法,这个方法和其他的线程控制方法(suspend,resume)一样都是过期了不建议使用的,这些方法都是不安全的。 例如stop()方法在结束一个线程的时候并不保证线程资源的正常释放,因此可能导致出现一些不确定的状态。 按照人类逻辑来理解:T1线程调用方法修改T2线程的状态,但是T2现在在做什么T1是不清楚的,所以强制他关闭就是不安全的,就好比在Linux中使用 kill -9 杀掉一个进程。

使用interrupt方法

interrupt()方法只是修改了被中断线程的中断标志 ,并没有做什么过分的事儿。就像平时写代码的时候修改某对象的标志,对象自己通过标志类决定执行什么逻辑。这里也是一样,interrupt()方法修改中断标志,被中断的线程,自己决定做什么事儿(中断或者不中断都是被中断线程自己决定的,外部只是通知他,不是强迫他)。追一下源码。

1.Java调用interrupt方法

2.通过指引找到 jvm.cpp#JVM_Interrupt方法

thread.cpp interrupt 借用操作系统。直接通过系统调用 interrupt
void Thread::interrupt(Thread* thread) {
  trace("interrupt", thread);
  debug_only(check_for_dangling_thread_pointer(thread);)
  // tips: 调用操作系统的interrupt方法
  os::interrupt(thread);
}

这里还是以os_linux.cpp为例最终调用osthread的set_interrupted修改状态

这里就印证了上方的 Thread.interrupt()只是修改了线程的一个标志位 ,并没有做什么过分的事儿。

线程的复位

interruptedisInterrupted

这两个放在一起是因为他们底层都是调用的同一个native方法isInterrupted()只是给了不同的入参。 再就是,有过面试官问到他两的区别,所以干脆放在一起。首先说结论 ,isInterrupted()会返回线程的中断状态,interrupted()不仅会返回中断状态,而且如果线程处于状态状态还会将线程终端状态复位(清除中断状态)。

os_linux.cpp的is_interrupted()方法印证了上面说的isInterrupted()会返回线程的中断状态,interrupted()不仅会返回中断状态,而且如果线程处于状态状态还会将线程终端状态复位(清除中断状态)。

其他的线程复位

Java中只要抛出了InnterruptException异常的方法都对线程进行了复位。先理顺下为什么要这么做:查看下基本上抛出InnterruptException异常的方法都是线程阻塞方法,比如sleep(),wait(),join()。这类方法执行后线程会处于TIMED_WAITING或者WAITING状态,处于这类状态的线程是不受控的(线程丧失了对自己的主导,需要其他的线程唤醒,或者阻塞时间到达才能拥有自己的主导权),这个时候线程中断,线程自己却没办法处理。甚至可能永远等不到释放而无法执行中断。所以,在线程是中断状态下,执行方法让线程阻塞,就要抛出一个异常告诉外界 ,我现在是阻塞状态,并且将中断标记复位,方便外界进行处理(例如中断线程的执行或者继续阻塞方法),相当于给了外界一个改变线程状态的入口。 以sleep()为例追踪下源码:

通过指引找到 jcm.cpp#JVM_Sleep

方法入口就直接判断线程的中断状态了 ,is_interrupted()上面介绍过了,参数为true就是清除中断标志并且返回清除之前的中断状态。这里线程是中断状态的就直接抛出 InnterruptException sleep interrupted异常了。

到此这篇关于Java线程生命周期的终止与复位的文章就介绍到这了,更多相关Java线程生命周期内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • java 线程的生命周期详解

    一个线程的生命周期: 新建状态: 使用 new 关键字和 Thread 类或其子类建立一个线程对象后,该线程对象就处于新建状态.它保持这个状态直到程序 start() 这个线程. 就绪状态: 当线程对象调用了start()方法之后,该线程就进入就绪状态.就绪状态的线程处于就绪队列中,要等待JVM里线程调度器的调度. 运行状态: 如果就绪状态的线程获取 CPU 资源,就可以执行 run(),此时线程便处于运行状态.处于运行状态的线程最为复杂,它可以变为阻塞状态.就绪状态和死亡状态. 阻塞状态: 如

  • Java 线程的生命周期详细介绍及实例代码

    当线程被创建并启动之后,它既不是一启动就进入执行状态,也不是一直处于执行状态,在其生命周期中,要经过"新建(New)"."就绪(Runnable)"."运行(Running')"."阻塞(Blocked)"和"死亡(Dead)"五种状态.线程在创建之后,不可能一直霸占着CPU独立运行,需要在多个线程之间切换,所以大部分时间处于运行.阻塞之间切换.  一.线程的状态 线程的存在有几种不同的状态,如下: New

  • 图解Java线程的生命周期

    在Java中,线程有5中不同状态,分别是:新建(New).就绪(Runable).运行(Running).阻塞(Blocked)和死亡(Dead).它们之间的转换图如下: 上图有一个例外,调用yield()方法可以让当前处于运行状态的线程转入就绪状态.如果要测试某线程是否已经死亡,可以使用isAlive()方法,该方法在线程处于就绪.运行.阻塞时放回true,新建和死亡时返回false.不要试图对一个已经死亡的线程调用start()方法而重新启动,死亡就是死亡和人一样,不可能再生.还有也不要对一

  • 初步学习Java中线程的实现与生命周期

    线程的实现 在Java中通过run方法为线程指明要完成的任务,有两种技术来为线程提供run方法: 1.继承Thread类并重写它的run方法.之后创建这个子类的对象并调用start()方法. 2.通过定义实现Runnable接口的类进而实现run方法.这个类的对象在创建Thread的时候作为参数被传入,然后调用start()方法. Thread类是专门用来创建线程和对线程进行操作的类.当某个类继承了Thread类之后,该类就叫做一个线程类. 两种方法均需执行线程的start()方法为线程分配必须

  • Java线程的生命周期和状态控制_动力节点Java学院整理

    一.线程的生命周期 线程状态转换图: 1.新建状态 用new关键字和Thread类或其子类建立一个线程对象后,该线程对象就处于新生状态.处于新生状态的线程有自己的内存空间,通过调用start方法进入就绪状态(runnable). 注意:不能对已经启动的线程再次调用start()方法,否则会出现Javalang.IllegalThreadStateException异常. 2.就绪状态 处于就绪状态的线程已经具备了运行条件,但还没有分配到CPU,处于线程就绪队列(尽管是采用队列形式,事实上,把它称

  • Java线程的生命周期的详解

    Java线程的生命周期的详解 对于多线程编程而言,理解线程的生命周期非常重要,本文就针对这一点进行讲解. 一.线程的状态 线程的存在有几种不同的状态,如下: New状态 Ready状态 Running状态 Dead状态 Non Runnable状态 1.New状态 New状态是线程已经被创建,但是还未开始运行的状态.此状态通过调用线程的start()方法可让线程运行. 2.Runnable状态 Runnable状态可称为准备运行状态,也可称为队列,此状态通过调用线程的start()方法可让线程运

  • java多线程编程之线程的生命周期

    复制代码 代码如下: // 开始线程public void start( );public void run( ); // 挂起和唤醒线程public void resume( ); // 不建议使用public void suspend( );// 不建议使用public static void sleep(long millis);public static void sleep(long millis, int nanos); // 终止线程public void stop( );   /

  • Java 线程的生命周期完整实例分析

    本文实例讲述了Java 线程的生命周期.分享给大家供大家参考,具体如下: 一 代码 /** * @Title: ThreadStatus.java * @Description: TODO(演示线程的生命状态) */ import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; import java.util.concurrent.TimeUnit; public class T

  • Java线程生命周期的终止与复位

    目录 Thread生命周期 生命周期概述 线程生命周期流程图 线程生命周期测试 启动线程 java中的启动 Hotspot中的启动 线程中断与复位 不要使用stop方法 使用interrupt方法 线程的复位 其他的线程复位 Thread生命周期 生命周期概述 Java的线程状态描述放在Thread类里面的枚举类State中.总共包含了6中状态(从出生到死亡). public enum State { /** * 尚未启动的线程的线程状态 (没有start) */ NEW, /** * 可运行线

  • Java线程生命周期及转换过程

    目录 Java 线程生命周期 生命周期转换 1.从 NEW 到 RUNNABLE 2.从 RUNNABLE 到 BLOCKED 3.从 RUNNABLE 到 WAITTING 4.从 RUNNABLE 到 TIMED_WATTING 5.RUNNABLE 到 TERMINATED 总结 前言: 线程的生命周期指的是线程从创建到销毁的整个过程,通常情况下线程的生命周期有以下 5 种: 初始状态 可运行状态 运行状态 休眠状态 终止状态 它们的状态转换如下图所示:  Java 线程生命周期 Java

  • Java中的线程生命周期核心概念

    目录 Java多线程 Java中线程的生命周期 NEW Runnable Blocked Waiting Timed Waiting Terminated 结论 前言: 在本文中,我们将详细讨论Java中的一个核心概念——线程的生命周期.我们将使用一个快速的图解,当然还有实用的代码片段来更好地理解线程执行期间的这些状态. Java多线程 在Java语言中,多线程是由线程的核心概念驱动的.线程在其生命周期中会经历各种状态: Java中线程的生命周期 java.lang.Thread类包含一个静态枚

  • java类型生命周期的详细解析

    开始阶段 装载:把二进制形式的java类型读入jvm中.1)通过该类型的完全限定名,产生一个代表该类型的二进制数据流:2)解析这个二进制数据流为方法区内的内部数据结构:3)创建一个表示该类型的java.lang.Class类的实例: 连接:把已读入的类型数据合并到虚拟机的运行时状态中.1)验证:确保java类型数据格式正确并且适用于jvm使用:2)准备:为该类型分配内存:3)解析:把常量池中的符号引用转换为直接引用: 初始化:每个类和接口在首次主动使用时初始化.为类变量赋予正确的初始值:1)如果

  • 通过代码实例解析JAVA类生命周期

    代码 public class TestClass { public static int k=0; public static TestClass t1=new TestClass("t1"); public static TestClass t2=new TestClass("t2"); public static int i=print("i"); public static int n=99; public int j=print(&qu

  • Java中Servlet的生命周期详解

    目录 Web基础和HTTP协议 什么是Servlet Servlet的生命周期 Web基础和HTTP协议 ┌─────────┐ ┌─────────┐ │░░░░░░░░░│ │O ░░░░░░░│ ├─────────┤ ├─────────┤ │░░░░░░░░░│ │ │ ├─────────┤ │ │ │░░░░░░░░░│ └─────────┘ └─────────┘ │ request 1 │ │─────────────────────>│ │ request 2 │ │───

  • Java 线程池详解及实例代码

    线程池的技术背景 在面向对象编程中,创建和销毁对象是很费时间的,因为创建一个对象要获取内存资源或者其它更多资源.在Java中更是如此,虚拟机将试图跟踪每一个对象,以便能够在对象销毁后进行垃圾回收. 所以提高服务程序效率的一个手段就是尽可能减少创建和销毁对象的次数,特别是一些很耗资源的对象创建和销毁.如何利用已有对象来服务就是一个需要解决的关键问题,其实这就是一些"池化资源"技术产生的原因. 例如Android中常见到的很多通用组件一般都离不开"池"的概念,如各种图片

  • Java 线程池详解及创建简单实例

    Java 线程池 最近在改进项目的并发功能,但开发起来磕磕碰碰的.看了好多资料,总算加深了认识.于是打算配合查看源代码,总结并发编程的原理. 准备从用得最多的线程池开始,围绕创建.执行.关闭认识线程池整个生命周期的实现原理.后续再研究原子变量.并发容器.阻塞队列.同步工具.锁等等主题.java.util.concurrent里的并发工具用起来不难,但不能仅仅会用,我们要read the fucking source code,哈哈.顺便说声,我用的JDK是1.8. Executor框架 Exec

  • Java线程的基本概念

    在之前的章节中,我们都是假设程序中只有一条执行流,程序从main方法的第一条语句逐条执行直到结束.从本节开始,我们讨论并发,在程序中创建线程来启动多条执行流,并发和线程是一个复杂的话题,本节,我们先来讨论Java中线程的一些基本概念. 创建线程 线程表示一条单独的执行流,它有自己的程序执行计数器,有自己的栈.下面,我们通过创建线程来对线程建立一个直观感受,在Java中创建线程有两种方式,一种是继承Thread,另外一种是实现Runnable接口,我们先来看第一种. 继承Thread Java中j

随机推荐