Java线程及线程池状态
一、Java线程的六种状态
- 如上图1,JDK定义线程状态是不存在“运行中”状态,但为方便描述过程有些图中会画出运行中的状态。
- Java线程创建后调用start方法进入就绪状态,被OS调度选中后运行,运行结束或程序退出或抛异常时终止。
- 运行中线程调用Thread.yield()方法状态切换为可运行就绪状态;运行中线程执行时遇到IO阻塞、调用Thread.sleep或其它线程对象join()方法时进入阻塞等待状态;运行中线程遇到同步锁方法时进入锁池等待状态。
- 线程从阻塞状态、锁池状态、新建状态、运行状态都可以在相关条件满足时回到可运行就绪状态,然后被OS调度选中进入运行状态。
二、“VisualVM线程监控线程状态”与“Java线程状态”对应关系
通过VisualVM监控JVM时,可以通过“线程”标签页查看JVM的线程信息,VisualVM的线程状态如下:
通过dump thread stack,并与VisualVM监控信息中的线程名称对应,找到的VisualVM每种线程状态的线程堆栈
三、Java线程池的五种状态
- 线程池的初始化状态是RUNNING,能够接收新任务,以及对已添加的任务进行处理。
- 线程池处在SHUTDOWN状态时,不接收新任务,但能处理已添加的任务。 调用线程池的shutdown()接口时,线程池由RUNNING -> SHUTDOWN。
- 线程池处在STOP状态时,不接收新任务,不处理已添加的任务,并且会中断正在处理的任务。 调用线程池的shutdownNow()接口时,线程池由(RUNNING or SHUTDOWN ) -> STOP。
- 当所有的任务已终止,ctl记录的”任务数量”为0,线程池会变为TIDYING状态。当线程池变为TIDYING状态时,会执行钩子函数terminated()。terminated()在ThreadPoolExecutor类中是空的,若用户想在线程池变为TIDYING时,进行相应的处理;可以通过重载terminated()函数来实现。
- 当线程池在SHUTDOWN状态下,阻塞队列为空并且线程池中执行的任务也为空时,就会由 SHUTDOWN -> TIDYING。 当线程池在STOP状态下,线程池中执行的任务为空时,就会由STOP -> TIDYING。
- 线程池彻底终止,就变成TERMINATED状态。线程池处在TIDYING状态时,执行完terminated()之后,就会由 TIDYING -> TERMINATED。