1 yield()介绍
yield()的作用是让步,他能让当前线程由"运行状态"到"就绪状态",从而让其他具有相同相同优先级的等待线程获取执行权,但是,并不能保证在当前线程调用yield()之后,其他具有相同优先级的线程就一定能获得执行权,也有可能又进入到"运行状态"继续运行"
2yield()示例
package com.tuhu.filt.javadatayield;
public class ThreadA extends Thread {
public ThreadA(String name){
super(name);
}
public synchronized void run(){
for(int i =0;i<10;i++){
System.out.printf("%s [%d]:%d\n", this.getName(), this.getPriority(), i);
if(i % 4 == 0){
Thread.yield();
}
}
}
}
class YieldTest{
public static void main(String[] args) {
ThreadA t1 = new ThreadA("t1");
ThreadA t2 = new ThreadA("t2");
t1.start();
t2.start();
}
}
我们可以看到在t2为0可以被4整除的时候,并没有切换到线程t1,这表明yield()虽然可以让线程由"运行状态转为就绪状态",但是它不一定会让其他线程获取CPU执行权(即,其他线程进入到"运行状态")即使这个"其他线程"与当前调用yield()的线程具有相同的优先级。
yield()与wait()的比较
我们知道wait()的作用是让当前线程由"运行状态"进入"等待(阻塞)状态"的同时,也会释放同步锁,而yield()的作用是让步,它也会让当前线程离开"运行状态"。他们的区别是
01 wait是会让线程由运行状态进入到等待(阻塞)状态的,而yield是让线程由运行状态进入到就绪状态
02 wait()是会让线程释放它所持有对象的同步锁,而yield()方法不会释放锁
下面通过示例演示yield()是不会释放锁的
package com.tuhu.filt.javadatayield;
public class YieldLockTest {
private static Object obj = new Object();
public static void main(String[] args) {
ThreadB t1 = new ThreadB("t1");
ThreadB t2 = new ThreadB("t2");
t1.start();
t2.start();
}
static class ThreadB extends Thread{
public ThreadB(String name){
super(name);
}
public void run(){
synchronized (obj){
for(int i = 0;i<10;i++){
System.out.printf("%s [%d]:%d\n", this.getName(), this.getPriority(), i);
// i整除4时,调用yield
if (i%4 == 0)
Thread.yield();
}
}
}
}
}
结果‘
可以看到上面代码线程对象是两个,但是有同步锁的却是类的对象,所以两个线程拿来拿去拿的都是这个类的同步锁,而且根据yield线程让步不会释放同步锁来看,结果完全符合这个特性,t1拿到同步锁之后就没释放,一直等到t1执行完才给了t2。