java中线程的状态

操作系统层面来叙述(五种)

在这里插入图片描述

  • 【初始状态】仅是在语言层面创建了线程对象,还未与操作系统线程关联
  • 【可运行状态】(就绪状态)指该线程已经被创建(与操作系统线程关联),可以由 CPU 调度执行
  • 【运行状态】指获取了 CPU 时间片运行中的状态
    • 当 CPU 时间片用完,会从【运行状态】转换至【可运行状态】,会导致线程的上下文切换
  • 【阻塞状态】
    • 如果调用了阻塞 API,如 BIO 读写文件,这时该线程实际不会用到 CPU,会导致线程上下文切换,进入【阻塞状态】
    • 等 BIO 操作完毕,会由操作系统唤醒阻塞的线程,转换至【可运行状态】
    • 与【可运行状态】的区别是,对【阻塞状态】的线程来说只要它们一直不唤醒,调度器就一直不会考虑调度它们
  • 【终止状态】表示线程已经执行完毕,生命周期已经结束,不会再转换为其它状态

java api层面来叙述(六种)

在这里插入图片描述

  • NEW 线程刚被创建,但是还没有调用 start() 方法
  • RUNNABLE 当调用了 start() 方法之后,注意,Java API 层面的 RUNNABLE 状态涵盖了 操作系统 层面的【可运行状态】,【运行状态】和【阻塞状态】(由于 BIO 导致的线程阻塞,在 Java 里无法区分,仍然认为是可运行)
  • BLOCKED , WAITING , TIMED_WAITING 都是 Java API 层面对【阻塞状态】的细分,后面会在状态转换一节
    详述
  • TERMINATED 当线程代码运行结束

下面对这6种状态进行解释:

  • 1.New(初始化状态):指的是在高级语言,比如Java中,在Java层面的线程被创建了,而在操作系统中的线程其实是还没被创建的,所以这个时候是不可能分配CPU执行这个线程的。所以这个状态是高级语言独有的,操作系统的线程没有这个状态。我们New 了一个线程,那时候它就是这个状态。
  • 2.Runnable(可运行/运行状态):这个状态下是可以分配CPU执行的,在New状态时候我们调用start()方法后线程就处于这个状态。
  • 3.Blocked(阻塞状态):这个状态下是不能分配CPU执行的,只有一种情况会导致线程阻塞,就是synchronized。我们知道被synchronized修饰的方法或者代码块同一时刻只能有一个线程执行,而其他竞争锁的线程就从Runnable到了Blocked状态!当某个线程竞争到了锁它就变成了Runnable状态。注意并发包中的Lock,是会让线程属于等待状态而不是阻塞,只有synchronized是阻塞。
  • 4.Waiting(无时间限制的等待状态):这个状态下也是不能分配CPU执行的。有三种情况会使得Runnable状态到Waiting状态:
    • 调用无参的Object.wait()方法。等到notifyAll()或者notify()唤醒就会回到Runnable状态。
    • 调用无参的Thread.join()方法。也就是比如你在主线程里面建立了一个线程A,调用A.join(),那么你的主线程是得等A执行完了才会继续执行,这时你的主线程就是等待状态。
    • 调用LockSupport.park()方法。LockSupport是Java6引入的一个工具类,Java并发包中的锁都是基于他实现的,在调用LockSupport.unpark(Thread thread),就会回到Runnable状态。
  • 5.Timed_Waiting(有时间限制的等待状态):其实这个状态和Waiting就是有没有超时时间差的差别,这个状态下也是不能分配CPU执行的。有五种情况会使得Runnable状态到Waiting状态
    • Object.wait(long timeout)
    • Thread.join(long millis)
    • Thread.sleep(long millis)(注意Thread.sleep(long millis,int nanos))内部调用的其实也是Thread.sleep(long milis).
    • LockSupport.parkNanos(Object blocked,long deadline)
    • LockSupport.parkUntil(long deadline)
  • 6.Terminated()终止状态:在我们的线程正常run结束之后或者run一半异常了就是终止状态。注意有个方法Thread.stop()是会让线程终止的,但是这个方法已经被弃用了,不推荐使用,因为比如你这个线程得到了锁,stop之后这个锁也没了,其他线程就都拿不到这个锁了。所以推荐使用interrupt()方法。
    • interrupt()会使得线程Waiting和Timed_Waiting状态的线程抛出interruptedException异常,使得Runnable状态的线层如果是在I/O操作会抛出其他异常
    • 如果Runnable状态的线程没有阻塞在I/O状态的话,那只能主动检测自己是不是被中断了,使用isInterrupted()。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值