yield()--让出CPU执行权
上一篇:sleep()–线程休眠
yield()–让出cpu执行权
yield()是Thread类下的一个静态方法
从注释中可以知道 这个方法一般用于测试和调试
《JAVA并发编程之美》书中是这么描述的:
“当一个线程调用yield()方法时,实际就是在暗示线程调度器当前线程请求让出自己的cpu使用,但是线程调度器可以无条件忽略这个暗示。操作系统是为每一个线程分配一个时间片来占有cpu,正常情况下当一个线程把时间片使用完之后线程调度器才会进行下一轮的线程调度,而当一个线程调用了yield()时,是在告诉线程调度器自己占有的时间片还没有使用完的部分自己不想使用了,这暗示线程调度器现在就可以进行下一轮的线程调度;当一个线程调用yield()方法时,当前线程会让出cpu的使用权,然后处于就绪状态,线程调度器会从线程就绪队列中获取一个线程优先级最高的线程,当然也可能会调度到刚刚调用yield()的线程来获取cpu的执行权”
总的来说就是,当一个线程调用了yield()方法后,线程调度器可以释放这个线程对cpu的执行权让其处于就绪状态,但线程调度器可以无条件忽视这个线程的暗示让这个线程一直占有着cpu;线程调用yield()方法后是出于就绪状态还是继续执行取决于线程调度器有没有执行新一轮的线程调度;
代码示例:
public class YieldTest {
public void test1(){
Thread thread1 = new Thread(()->{
System.out.println("线程1 执行当中");
System.out.println("线程1 暗示线程,可以进行新一轮的线程调度");
Thread.yield();
System.out.println("线程1 放弃cpu执行权");
});
thread1.start();
}
public void test2(){
Thread thread1 = new Thread(()->{
System.out.println("线程2 执行当中");
System.out.println("线程2 暗示线程,可以进行新一轮的线程调度");
Thread.yield();
System.out.println("线程2 放弃cpu执行权");
});
thread1.start();
}
public static void main(String[] args) {
YieldTest yieldTest = new YieldTest();
yieldTest.test1();
yieldTest.test2();
}
}
多次执行结果:
可以看到
第一次执行的时候线程1在调用了yield()方法后还是执行结束了 说明线程1在调用yield()方法后没有放弃cpu的执行权,即线程调度器没有进行新一轮的调度;
第二次执行的时候当线程1调用yield()后,线程2开始执行,说明在线程1调用yield()方法后放弃了cpu的执行权,即线程调度器开始了新一轮的调度,然后在线程2执行完成之后线程1也执行完毕,说明线程在调用yield()方法后不是处于阻塞状态而是就绪状态,如果是处于阻塞状态的话在线程2执行完毕之后线程1是不会继续执行的;
总结:线程调用yield()方法后时继续执行还是等待下一轮调度,取决于线程调度器是否释放当前线程的cpu执行权,而这个过程是随机的,正是因为这个不确定性所以一般很少使用这个方法;在调试或者测试的时候这个方法或许可以帮助复现由于并发竞争条件导致的问题,在设计并发控制时或许可以使用到。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南