前言
Java 多线程中的方法有很多,我们就挑几个有代表性的来分析
正文
sleep(睡眠)
- 调用 sleep 会让当前线程从 Running --》Timed Waiting 状态(阻塞)
- 其他线程使用 interrupt 方法打断正在睡眠的线程, sleep 方法会抛出 InterruptedException PS: 好比一个在睡觉的人,被吵醒,有起床气!!!
- 睡眠结束后的线程未必会立刻得到执行,因为这时的CPU 可能正在执行其他代码
- JDK 1.5 之后增加了 TimeUnit 的sleep 代替 Thread 的 sleep 来获得更好的可读性
应用
可以使用 sleep 让 while(true) 的代码让出 CPU 的使用权
public static void main(String[] args) {
while(true){
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
- 可用 wait 或条件变量达到类似的效果,但需要加锁,并且需要唤醒,适用进行同步的场景
- sleep 适用于无需锁同步的场景
yield(谦让,让出)
- 调用 yield 会让当前线程Running --》 Runnabel 状态(就绪),接着调用执行其他同优先级的线程。如果这时没有同优先级的线程,则不能保证让当前线程暂停的效果。本质就是让出 CPU 时间片
- 具体的实现依赖于操作系统的任务调度器
阻塞状态与就绪状态比较
- 就绪状态有机会得到 CPU 的时间片,而阻塞状态的线程,任务调度器是不考虑分时间片给它的,除非等待等待时间结束。
- sleep 是有一段时间段的休眠,而 yield 是一次性的某个时间点让出 CPU
线程优先级
-
线程优先级会提示(hint)调度器优先调度该线程,但只是一个提示,任务调度器可以忽略它
-
若 CPU 较忙,优先级高的线程会获得更多时间片;CPU 闲时,优先级几乎没作用
Join 方法
join() 等待线程运行结束
谁调用 join ,那么就等谁运行结束
同步?
以调用角度来看
需要等结果返回,才能继续运行的就是同步
不需要等待结果返回,就能继续运行就是异步
interrupt
可打断 sleep、wait、join 的线程
两阶段终止模式
概念
一个线程 T1 用这种模式体面的终止T2 线程
毕竟 T2 是个体面人,让自己体面体面
不推荐
- 使用 stop() 方法停止线程, 若 线程此时锁住了资源,被杀死后则会无法释放锁
- 使用 System.exit(int) 方法停止线程,会导致整个程序都停止
分析