Java之线程的状态转换

一、操作系统层面的五种状态:

  • 初始状态:仅是在语言层面创建了线程对象,还未与操作系统线程关联。

  • 可运行状态:指该线程已经被创建(与操作系统线程关联),可以由CPU调度执行

  • 运行状态:指获取了CPU时间片运行中的状态

    • 当CPU时间片用完,会从运行状态转至可运行状态,回导致线程上下文切换

  • 阻塞状态:

    • 如果调用了阻塞API,如果BIO读写文件,这时该线程实际不会用到CPU,会导致线程上下文切换,进入阻塞状态

    • 等BIO操作完毕,会由操作系统唤醒阻塞的线程,转换至可运行状态

    • 与可运行状态的区别是,对阻塞状态的线程来说只要他们一直不唤醒,调度器就一直不会考虑调度他们。

  • 终止状态:表示现场已经执行完毕,生命周期已经结束,不会再转换为其他状态。

二、JavaAPI层面的六种线程的状态

NEW、RUNNABLE、BLOCKED、WAITING、TIMED_WAITING、TERMINATED。

NEW状态表示刚刚创建的进程,这种线程还没开始执行,还未和操作系统层面的线程关联,等到线程的start方法调用时,线程才开始执行。

当线程执行时,处于RUNNABLE状态(Java API层面的Runnable状态涵盖了操作系统层面的可运行状态、运行状态、阻塞状态,由于BIO导致的线程阻塞再Java里无法区分,任然认为是可运行的),表示线程所需的一切资源都已经准备好了。

如果线程在执行过程中遇到了synchronized同步块,就会进入BLOCKED阻塞状态,这时线程就会暂停执行,直到获得请求的锁。

WAITING和TIMED_WAITING都表示等待状态,它们的区别是WAITING会进入一个无时间限制的等待,TIMED_WAITING会进行一个有时限的等待。wait()方法等待的线程在等待notify()方法,而通过join()方法等待的线程会等待目标线程的终止。

状态转换图

假设有线程Thread t

情况1 NEW-->RUNNABLE:

  • 当调用t.start()方法时,由NEW-->RUNNABLE

情况2 RUNNABLE<-->WAITING:t线程调用synchronized(obj)获取了对象锁后

  • 调用obj.wait()方法时,t线程从RUNNABLE-->WAITING

  • 在上一点的基础上,其他线程调用obj.notify(),obj.notifyAll(),t.interrupt()时

    • 竞争锁成功,t线程从WAITING-->BLOCKED-->RUNNABLE

    • 竞争锁失败,t线程从WAITING-->BLOCKED

情况3 RUNNABLE<-->WAITING:

  • 当前线程调用t.join()方法时,当前线程从RUNNABLE-->WAITING

    • 注意当前线程实在t线程对象的监视器上等待

  • t线程运行结束,或调用了当前线程的interrupt()时,当前线程从WAITING-->RUNNABLE

情况4 RUNNABLE<-->WAITING:

  • 当前线程调用LockSupport.park()方法会让当前线程从RUNNABLE-->WAITING

  • 调用LockSupport.unpark(目标线程)或调用目标线程的interrupt()时,会让目标线程从WAITING-->RUNNABLE

情况5 RUNNABLE<-->TIMED_WAITING:t线程调用synchronized(obj)获取了对象锁后

  • 调用obj.wait(long n)方法时,t线程从RUNNABLE-->TIMED_WAITING

  • 在上一点的基础上,t线程等待时间超过了n毫秒,或其他线程调用obj.notify(),obj.notifyAll(),t.interrupt()时

    • 竞争锁成功,t线程从WAITING-->BLOCKED-->RUNNABLE

    • 竞争锁失败,t线程从WAITING-->BLOCKED-->BLOCKED

情况6 RUNNABLE<-->TIMED_WAITING:

  • 当前线程调用t.join(long n)方法时,当前线程从RUNNABLE-->TIMED_WAITING

    • 注意当前线程实在t线程对象的监视器上等待

  • 当前线程等待时间超过了n毫秒,或t线程运行结束,或调用了当前线程的interrupt()时,当前线程TIMED_WAITING-->RUNNABLE

情况7 RUNNABLE<-->TIMED_WAITING:

  • 当前线程调用Thread.sleep(long n),当前线程从RUNNABLE-->TIMED_WAITING

  • 当前线程等待时间超过了n毫秒,当前线程从TIMED_WAITING-->RUNNABLE

情况8 RUNNABLE<-->TIMED_WAITING:

  • 当前线程调用了LockSupport.parkNanos(long nanos)或LockSupport.parkUntil(long millis)时,当前线程从RUNNABLE-->TIMED_WAITING

  • 调用LockSupport.unpark(目标线程)或调用了线程的interrupt(),或是等待超时,会让目标现场从TIMED_WAITING-->RUNNBALE

情况9 RUNNABLE<-->BLOCKED:

  • t线程用synchronized(obj)获取对象锁时如果竞争失败,从RUNNABLE-->BLOCKED

  • 持有obj锁的线程同步代码块执行完毕,会唤醒该对象上所有BLOCKED对象重新竞争,如果其中t线程竞争成功,从BLOCKED-->RUNNABLE,其他失败的线程仍然BLOCKED

情况10 RUNNABLE<-->TERMINATED

当前线程所有代码运行完毕,进入TERMINATED

注意:从NEW状态出发后,线程不能再回到NEW状态,同理,处于TERMINATED的线程也不能再回到RUNNABLE状态。

  • 3
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值