前言:现在的操作系统在运行一个程序时, 会为其创建一个进程。 在进程中操作系统调度的最小单元是线程, 也叫轻量级进程(Light Weight Process) , 在一个进程里可以创建多个线程, 这些线程都拥有各自的计数器、 堆栈和局部变量等属性, 并且能够访问共享的内存变量。 处理器在这些线程上高速切换, 让使用者感觉到这些线程在同时执行。
啰嗦一下进程、线程是什么?
进程是指一个内存中运行的应用程序,每个进程都有自己独立的一块内存空间,一个进程中可以启动多个线程。比如在Windows系统中,一个运行的exe就是一个进程。
线程是指进程中的一个执行流程,一个进程中可以运行多个线程。比如java.exe进程中可以运行很多线程。线程总是属于某个进程,进程中的多个线程共享进程的内存。
线程的生命周期
每个线程都有自己的完整的生命周期,下面通过一张copy的图来详细的了解一下:
从上图我们可以看出线程的生命周期大致可以分为五个阶段:
NEW(新建状态)
当我们new一个Thread对象时,此时它并不处于运行状态,因为还没有调用start方法启动线程。那么线程的NEW状态,其实只是Thread对象的状态,在没有调用start方法之前,该线程根本不存在,和new一个普通的Java对象没什么区别。NEW状态可以通过start方法进入RUNNABLE状态。
RUNNABLE(就绪状态)
线程对象进入RUNNABLE状态必须调用start方法,此时JVM进程中才会真正的创建一个线程,线程启动后并不会立即得到执行。线程是否运行和进程一样都要听从CPU的调度,为此我们把这个中间状态成为就绪状态,也称为可执行状态(RUNNABLE),也就是说它具备执行的资格,但是并没有真正的执行而是在等待CPU的调度。
由于存在Running状态,所以不会直接进人BLOCKED状态和TERMINATED状态,即使是在线程的执行逻辑中调用wait、sleep或者其他block的I0操作等,也必须先获得CPU的调度执行权才可以,严格来讲,RUNNABLE的线程只能意外终止或者进人RUNNING状态。
RUNNING(运行状态)
一旦CPU通过时间片轮转或者其他方式选中了线程,那么此时它才能真正的执行自己的逻辑。这里需要注意的一点是一个正在RUNNING状态的线程其实也是RUNNABLE的,但是反过来则不成立。
在RUNNING状态中,线程的状态可以发生如下的状态转换:
- 直接进人TERMINATED状态,比如调用JDK已经不推荐使用的stop方法或者判断某个逻辑标识。
- 进人BLOCKED状态,比如调用了sleep,或者wait方法而加入了waitSet 中。
- 进行某个阻塞的I0操作,比如因网络数据的读写而进入了BLOCKED状态。