(二)Java线程与系统线程和生命周期
# 1.操作系统中线程的5种状态
在操作系统中,线程的生命周期可以分为5种状态:
①new 关键字创建了Thread类(或其子类)的对象,或者Runnable。
②Runnable 调用了start()方法,这时的线程就等待时间片轮转到自己这,以便获得CPU;第二种情况是线程在处于RUNNING状态时并没有运行完自己的run方法,时间片用完之后回到RUNNABLE状态;还有种情况就是处于BLOCKED状态的线程结束了当前的BLOCKED状态之后重新回到RUNNABLE状态。
③Running:这时的线程指的是获得CPU的RUNNABLE线程,RUNNING状态是所有线程都希望获得的状态。
④Dead:处于RUNNING状态的线程,在执行完run方法之后,就变成了DEAD状态了。
⑤Blocked:这种状态指的是处于RUNNING状态的线程,出于某种原因,比如调用了sleep方法、等待返回(调用wait方法)。
# 2. Java中线程的6种状态
在《Java并发编程》、《Java核心技术卷一》这两本书说到:在Java中,线程是分6种状态的。
Thread类的枚举也印证了这点:
状态名称 | 说明 |
---|---|
NEW | 初始状态,线程刚被构建,但是还没有调用start()方法 |
RUNNABLE | 运行状态,Java系统系统中将操作系统中的可运行状态 和运行状态 统称为RUNNABLE |
BLOCKED | 阻塞状态,表示线程阻塞于锁 |
WAITTING | 等待状态,表示线程进入等待状态,进入该状态表示当前线程做出一些特定动作(通知或者中断) |
TIME_WAITTING | 超时等待状态,该状态不同于等待状态,它可以在指定的时间后自行返回 |
TERMINATED | 中止状态,表示当前线程已经执行完毕 |
Blocked也分三种:
(一). 等待阻塞:运行(running)的线程执行o.wait()方法,JVM会把该线程放入等待队列(waitting queue)中,通过调用motify()方法回到就绪状态。
(二). 同步阻塞:运行(running)的线程在获取对象的同步锁时,若该同步锁被别的线程占用,则JVM会把该线程放入锁池(lock pool)中。
(三). 其他阻塞:运行(running)的线程执行Thread.sleep(long ms)或t.join()方法,或者发出了I/O请求时,JVM会把该线程置为阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新自动转入可运行(runnable)状态。
以下一小段来自于:这么说线程生命周期是不是简单了点? (opens new window)
其中Java中:
- 将通用线程状态的
可运行状态
和运行状态
合并为Runnable
, - 将
休眠状态
细分为三种 (BLOCKED
/WAITING
/TIMED_WAITING
); 反过来理解这句话,就是这三种状态在操作系统的眼中都是休眠状态,同样不会获得CPU使用权
# NEW
Thread myThread = new MyThread();
NEW 表示线程创建了,但是还没有开始执行。
# RUNNABLE
Runnable表示线程正在可执行状态。包括运行状态和可运行状态两种。
myThread.start();
当执行.start()
表示在可运行状态,拿到了CPU资源表示运行状态,所以为什么Runnable
会包括运行状态和可运行状态。
# BLOCKED
BLOCKED表示线程正在等待资源锁,而目前该资源正在被其他线程占有。
等待用户输入、synchronized锁等待都会造成阻塞。
# WAITING
WAITING 状态表示线程正在等待其他的线程执行特定的操作。有三种方法可以导致线程处于WAITTING状态:
- object.wait()
- thread.join()
- LockSupport.park()
# TIMED_WAITING
TIMED_WAITING状态表示在一个有限的时间内等待其他线程执行特定的某些操作。
java中有5种方式来达到这种状态:
- thread.sleep(long millis)
- wait(int timeout) 或者 wait(int timeout, int nanos)
- thread.join(long millis)
- LockSupport.parkNanos
- LockSupport.parkUntil
TIMED_WAITING和WAITTING有什么区别呢?
TIMED_WAITING
如果在给定的时间内没有等到其他线程的特定操作,则会被唤醒,从而进入争夺资源锁的队列,如果能够获取到锁,则会变成Runnable
状态,如果获取不到锁,则会变成BLOCKED
状态。
# TERMINATED
表示线程已经执行完毕,run()
方法执行结束则线程处于TERMINATED
状态。
参考:
这么说线程生命周期是不是简单了点? (opens new window)
https://www.cnblogs.com/wxd0108/p/5479442.html (opens new window)
https://www.cnblogs.com/developer_chan/p/10391365.html (opens new window)
https://www.cnblogs.com/flydean/p/12680277.html (opens new window)