java中线程的生命周期

Wesley13
• 阅读 784

java中线程的生命周期

线程是java中绕不过去的一个话题, 今天本文将会详细讲解java中线程的生命周期,希望可以给大家一些启发。

java中Thread的状态

java中Thread有6种状态,分别是:

  1. NEW - 新创建的Thread,还没有开始执行

  2. RUNNABLE - 可运行状态的Thread,包括准备运行和正在运行的。

  3. BLOCKED - 正在等待资源锁的线程

  4. WAITING - 正在无限期等待其他线程来执行某个特定操作

  5. TIMED_WAITING - 在一定的时间内等待其他线程来执行某个特定操作

  6. TERMINATED - 线程执行完毕

我们可以用一个图来直观的表示:

java中线程的生命周期

image

JDK代码中的定义如下:

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;    }

NEW

NEW 表示线程创建了,但是还没有开始执行。我们看一个NEW的例子:

public class NewThread implements Runnable{    public static void main(String[] args) {        Runnable runnable = new NewThread();        Thread t = new Thread(runnable);        log.info(t.getState().toString());    }    @Override    public void run() {    }}

上面的代码将会输出:

NEW

Runnable

Runnable表示线程正在可执行状态。包括正在运行和准备运行两种。

为什么这两种都叫做Runnable呢?我们知道在多任务环境中,CPU的个数是有限的,所以任务都是轮循占有CPU来处理的,JVM中的线程调度器会为每个线程分配特定的执行时间,当执行时间结束后,线程调度器将会释放CPU,以供其他的Runnable线程执行。

我们看一个Runnable的例子:

public class RunnableThread implements Runnable {    @Override    public void run() {    }    public static void main(String[] args) {        Runnable runnable = new RunnableThread();        Thread t = new Thread(runnable);        t.start();        log.info(t.getState().toString());    }}

上面的代码将会输出:

RUNNABLE

BLOCKED

BLOCKED表示线程正在等待资源锁,而目前该资源正在被其他线程占有。

我们举个例子:

public class BlockThread implements Runnable {    @Override    public void run() {        loopResource();    }    public static synchronized void loopResource() {        while(true) {            //无限循环        }    }    public static void main(String[] args) throws InterruptedException {        Thread t1 = new Thread(new BlockThread());        Thread t2 = new Thread(new BlockThread());        t1.start();        t2.start();        Thread.sleep(1000);        log.info(t1.getState().toString());        log.info(t2.getState().toString());        System.exit(0);    }}

上面的例子中,由于t1是无限循环,将会一直占有资源锁,导致t2无法获取资源锁,从而位于BLOCKED状态。

我们会得到如下结果:

12:40:11.710 [main] INFO com.flydean.BlockThread - RUNNABLE12:40:11.713 [main] INFO com.flydean.BlockThread - BLOCKED

WAITING

WAITING 状态表示线程正在等待其他的线程执行特定的操作。有三种方法可以导致线程处于WAITTING状态:

  1. object.wait()

  2. thread.join()

  3. LockSupport.park()

其中1,2方法不需要传入时间参数。

我们看下使用的例子:

public class WaitThread implements  Runnable{    public static Thread t1;    @Override    public void run() {        Thread t2 = new Thread(()->{            try {                Thread.sleep(10000);            } catch (InterruptedException e) {                Thread.currentThread().interrupt();                log.error("Thread interrupted", e);            }            log.info("t1"+t1.getState().toString());        });        t2.start();        try {            t2.join();        } catch (InterruptedException e) {            Thread.currentThread().interrupt();            log.error("Thread interrupted", e);        }        log.info("t2"+t2.getState().toString());    }    public static void main(String[] args) {        t1 = new Thread(new WaitThread());        t1.start();    }}

在这个例子中,我们调用的t2.join(),这会使调用它的t1线程处于WAITTING状态。

我们看下输出结果:

12:44:12.958 [Thread-1] INFO com.flydean.WaitThread - t1 WAITING12:44:12.964 [Thread-0] INFO com.flydean.WaitThread - t2 TERMINATED

TIMED_WAITING

TIMED_WAITING状态表示在一个有限的时间内等待其他线程执行特定的某些操作。

java中有5中方式来达到这种状态:

  1. thread.sleep(long millis)

  2. wait(int timeout) 或者 wait(int timeout, int nanos)

  3. thread.join(long millis)

  4. LockSupport.parkNanos

  5. LockSupport.parkUntil

我们举个例子:

public class TimedWaitThread implements  Runnable{    @Override    public void run() {        try {            Thread.sleep(5000);        } catch (InterruptedException e) {            Thread.currentThread().interrupt();            log.error("Thread interrupted", e);        }    }    public static void main(String[] args) throws InterruptedException {        TimedWaitThread obj1 = new TimedWaitThread();        Thread t1 = new Thread(obj1);        t1.start();        // The following sleep will give enough time for ThreadScheduler        // to start processing of thread t1        Thread.sleep(1000);        log.info(t1.getState().toString());    }}

上面的例子中我们调用了Thread.sleep(5000)来让线程处于TIMED_WAITING状态。

看下输出:

12:58:02.706 [main] INFO com.flydean.TimedWaitThread - TIMED_WAITING

那么问题来了,TIMED_WAITING和WAITTING有什么区别呢?

TIMED_WAITING如果在给定的时间内没有等到其他线程的特定操作,则会被唤醒,从而进入争夺资源锁的队列,如果能够获取到锁,则会变成Runnable状态,如果获取不到锁,则会变成BLOCKED状态。

TERMINATED

TERMINATED表示线程已经执行完毕。我们看下例子:

public class TerminatedThread implements Runnable{    @Override    public void run() {    }    public static void main(String[] args) throws InterruptedException {        Thread t1 = new Thread(new TerminatedThread());        t1.start();        // The following sleep method will give enough time for        // thread t1 to complete        Thread.sleep(1000);        log.info(t1.getState().toString());    }}

输出结果:

13:02:38.868 [main] INFO com.flydean.TerminatedThread - TERMINATED

本文的例子可以参考https://github.com/ddean2009/learn-java-concurrency/tree/master/thread-lifecycle

更多教程请参考 flydean的博客

本文分享自微信公众号 - 程序那些事(flydean-tech)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
待兔 待兔
6个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Wesley13 Wesley13
3年前
java多线程和异步回调
   在实际开发过程中遇到的多线程情况不多,但是在生产环境中多线程是最基本的情况,java面试时也会考到,所以看看多线程的知识还是很有必要的。 Thread,Runnable,Callable,Future,FutureTask,Executors这是java常见的接口和类。  thread.run():线程具体要执行的代码,thread.jo
浪人 浪人
3年前
一篇文章弄懂Java多线程基础和Java内存模型
文章目录一、多线程的生命周期及五种基本状态二、Java多线程的创建及启动1.继承Thread类,重写该类的run()方法2.通过实现Runnable接口创建线程类3.通过Callable和Future接口创建线程三、Java内存模型概念四、内存间的交互操作五、volatile和synchronized的
Wesley13 Wesley13
3年前
Java线程的6种状态及切换(透彻讲解)
Java中线程的状态分为6种。1\.初始(NEW):新创建了一个线程对象,但还没有调用start()方法。2\.运行(RUNNABLE):Java线程中将就绪(ready)和运行中(running)两种状态笼统的称为“运行”。线程对象创建后,其他线程(比如main线程)调用了该对象的start()方法。该状态的
Wesley13 Wesley13
3年前
4、jstack查看线程栈信息
1、介绍利用jps、top、jstack命令找到进程中耗时最大的线程,以及线程状态等等,同时最后还可以显示出死锁的线程查找:FoundoneJavaleveldeadlock即可1、jps获得进程号!(https://oscimg.oschina.net/oscnet/da00a309fa6
Wesley13 Wesley13
3年前
Java日期时间API系列31
  时间戳是指格林威治时间1970年01月01日00时00分00秒起至现在的总毫秒数,是所有时间的基础,其他时间可以通过时间戳转换得到。Java中本来已经有相关获取时间戳的方法,Java8后增加新的类Instant等专用于处理时间戳问题。 1获取时间戳的方法和性能对比1.1获取时间戳方法Java8以前
Wesley13 Wesley13
3年前
JAVA优化篇 如何从茫茫日志中找到运行缓慢的线程
引入  JAVA提供了一些分析DUMP的工具,比如jmap,visualvm等  JAVA还有寻找线程状态的工具,jstack等  数据库也有检查连接数,连接状态的命令,status,processlist等  代码中也可以添加一些时间的信息,对比信息发现可优化的地方  但这些都不是今天要记录的内容,今天要做的是使用一个比较
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Wesley13 Wesley13
3年前
Java多线程学习笔记
Java中创建多线程的三种方法1、继承Thread类创建线程2、实现Runnable接口创建线程3、使用Callable和Future创建线程\