网站首页 > java教程 正文
线程状态介绍
这里我们讲的是Java中的线程状态。
线程状态如下:
“线程初始状态:NEW线程运行状态:RUNNABLE线程阻塞状态:BLOCKED线程等待状态:WAITING超时等待状态:TIMED_WAITING线程终止状态:TERMINATED”
其中等待状态应该是一个比较复杂且重要的状态。
线程进入等待状态,即线程因为某种原因放弃了CPU使用权,阻塞也分为几种情况:
- 等待阻塞:运行的线程执行wait方法,JVM会把当前线程放入到等待队列
- 同步阻塞:运行的线程在获取对象的同步锁时,若该同步锁被其他线程锁占用了,那么JVM会把当前的线程放入到锁池中
- 其他阻塞:运行的线程执行Thread.sleep或者join方法,或者发出了I/O请求时,JVM会把当前线程设置为阻塞状态,当sleep结束join线程终止、I/O处理完毕则线程恢复
线程状态间的转换如下图:
下面我将讲解哪些情况会进入这些状态。
线程sleep时的状态
运行结果
new t1 t1 的状态:NEW
t1 running is false,t1将sleep
t1.sleep()时的状态:TIMED_WAITING
我们来分析一下,当new Thread时,线程t1[Thread-0]状态为NEW。线程启动,执行run()方法,打印t1 running is false,t1将sleep,此时线程t1睡眠。然后主线程睡眠,变量running设置为false。这线程t1还在睡眠中。再将主线程睡眠,线程t1仍然在睡眠中。此时线程t1的状态为TIMED_WAITING。
如果我们将线程t1中的睡眠时间修改一下
Thread.sleep(10000L); -> Thread.sleep(3000L);
再来看看运行结果
new t1 t1 的状态:NEW
t1 running is false,t1将sleep
t1.sleep()时的状态:TERMINATED
线程t1终止,所以说看代码时不要认为一看到sleep()就是超时等待【TIMED_WAITING】状态。要看看线程是否已经终止了。
线程join时的状态
运行结果如下
t2中执行t1.join(5000L)
t2的状态:TIMED_WAITING
t2中执行t1.join()
t2的状态:WAITING
t2执行完
当执行
t1.start();
t2.start();
线程t1执行睡眠,此时进入线程t2,并执行t2中执行t1.join(5000L)。接下来t1会抢占,进入主线程,主线程睡眠,此时t2还在等待t1,所以t2的线程状态为【TIMED_WAITING】。这时主线程又睡眠,t2里面开始执行t1.join(),此时t2的状态为【WAITING】。
线程synchronized时的状态
我们来看运行结果
t1抢不到锁的状态:BLOCKED
t1抢到锁
主线程启动,先抢到锁。此时t1.start()启动了t1线程,这时候主线程睡眠,锁还没有释放。此时的t1状态为【BLOCKED】。
线程wait时的状态
运行结果
t1将wait(1000L)
t1的状态:TIMED_WAITING
t1的状态:BLOCKED
t1将wait
t1的状态:WAITING
t1将执行完
t1的状态:RUNNABLE
t1的状态:TERMINATED
主线程启动,执行t1.start(),进入t1,执行t1将wait(1000L),此时t1让出锁。在t1超时等待的同时,主线程睡眠。当后,这里主线程抢到锁,t1的状态为【TIMED_WAITING】。这时主线程执行object.notify(),但这时锁还没有释放,t1还没有获取到锁,所以t1状态【BLOCKED】。之后主线程释放锁,t1获得锁,执行object.wait(),这时t1的状态【WAITING】。然后回到主线程,并获得锁,执行object.notify()。此时t1线程被唤醒并处于运行状态【RUNNABLE】。t1执行完成,状态为【TERMINATED】。
线程park()时的状态
t1 park后的状态:WAITING
t1 unpark后的状态:WAITING
大家可以思考下线程t1为什么是这个状态,有机会我会写一篇LockSupport的文章。
猜你喜欢
- 2024-12-12 Java 应用性能瓶颈剖析与多线程优化实战
- 2024-12-12 面试突击35:如何判断线程池已经执行完所有任务了?
- 2024-12-12 吞吐下降、RT增长、CPU飚高,都是 线程状态惹的祸?
- 2024-12-12 一文搞懂Java多线程
- 2024-12-12 Java线程池的关闭
- 2024-12-12 你能说出多线程中sleep、yield、join的用法及sleep与wait区别?
- 2024-12-12 队列满了、任务还在提交,看看Java线程池的任务饱和策略
- 2024-12-12 「重磅开篇」形成完善的多线程世界观
- 2024-12-12 一文深入理解java中的线程
- 2024-12-12 全局视角看技术-Java多线程演进史
你 发表评论:
欢迎- 最近发表
-
- pyinstaller打包python程序高级技巧
- 将python打包成exe的方式(python打包成exe的方法)
- Python打包:如何将 Flask 项目打包成exe程序
- py2exe实现python文件打包为.exe可执行程序(上篇)
- 如何将 Python 项目打包成 exe,另带卸载功能!
- Python打包成 exe,太大了该怎么解决?
- 可视化 Python 打包 exe,这个神器绝了!
- 案例详解pyinstaller将python程序打包为可执行文件exe
- Cocos 3.x 菜鸟一起玩:打包window程序
- 怎么把 Python + Flet 开发的程序,打包为 exe ?这个方法很简单!
- 标签列表
-
- java反编译工具 (77)
- java反射 (57)
- java接口 (61)
- java随机数 (63)
- java7下载 (59)
- java数据结构 (61)
- java 三目运算符 (65)
- java对象转map (63)
- Java继承 (69)
- java字符串替换 (60)
- 快速排序java (59)
- java并发编程 (58)
- java api文档 (60)
- centos安装java (57)
- java调用webservice接口 (61)
- java深拷贝 (61)
- 工厂模式java (59)
- java代理模式 (59)
- java.lang (57)
- java连接mysql数据库 (67)
- java重载 (68)
- java 循环语句 (66)
- java反序列化 (58)
- java时间函数 (60)
- java是值传递还是引用传递 (62)
本文暂时没有评论,来添加一个吧(●'◡'●)