Java線程Thread的yeild方法可能在日常使用中比較少出現(xiàn),那它是做什么用的谴供,我們先查查官方文檔解釋。
yield
public static void yield()
Causes the currently executing thread object to temporarily pause and allow other threads to execute.
看原版官方文檔可以避免出現(xiàn)歧義癣蟋,我的理解是
造成當前正在執(zhí)行的線程對象臨時性的暫停,和允許其他線程去執(zhí)行
也就是說狰闪,調(diào)用了yield方法之后疯搅,當前線程會臨時性的暫停一下,然后其他線程有機會去執(zhí)行任務(wù)埋泵。那么實際情況是怎么樣幔欧,我們用例子來測試一下。
public class YieldTest{
private static final int DEST_NUM = 50;
public static void main(String[] args) {
new YieldTest();
}
public YieldTest(){
ThreadDemo yt1 = new ThreadDemo("張三");
ThreadDemo yt2 = new ThreadDemo("李四");
ThreadDemo yt3 = new ThreadDemo("王五");
//yt1.setPriority(10);
//yt1.setPriority(5);
//yt3.setPriority(1);
yt1.start();
yt2.start();
yt3.start();
}
public class ThreadDemo extends Thread{
public ThreadDemo(String name){
setName(name);
}
@Override public void run() {
for (int i = 1; i <= DEST_NUM; i++) {
// 當i為30時丽声,該線程就會把CPU時間讓掉礁蔗,讓其他或者自己的線程執(zhí)行(也就是誰先搶到誰執(zhí)行)
if (i % 5 == 0) {
System.out.println("" + this.getName() + "-----" + i + " yeild一下");
yield();
}else{
System.out.println("" + this.getName() + "-----" + i);
}
}
}
}
}
上面的例子是,開啟張三雁社,李四浴井,王五三個線程分別打印從1到50的數(shù)字,每當打印的數(shù)字是5的倍數(shù)時(比如打印5,10,15等)霉撵,就yield一下磺浙。那么我們看看運行之后的結(jié)果是怎么樣的。
張三-----1
王五-----1
李四-----1
王五-----2
王五-----3
王五-----4
張三-----2
張三-----3
王五-----5 yeild一下
李四-----2
李四-----3
李四-----4
李四-----5 yeild一下 /////////
李四-----6
李四-----7
李四-----8
王五-----6
王五-----7
王五-----8
王五-----9
王五-----10 yeild一下
張三-----4
張三-----5 yeild一下 //////////
張三-----6
張三-----7
張三-----8
張三-----9
張三-----10 yeild一下 /////////
張三-----11
張三-----12
張三-----13
張三-----14
張三-----15 yeild一下 /////////
張三-----16
張三-----17
張三-----18
張三-----19
//其余的省略
徒坡。屠缭。。
可以看到這次張三雖然多次yeild了崭参,但是依然繼續(xù)執(zhí)行了。我們再運行一次看看
張三-----1
張三-----2
張三-----3
張三-----4
張三-----5 yeild一下
李四-----1
李四-----2
李四-----3
李四-----4
李四-----5 yeild一下
王五-----1
王五-----2
王五-----3
王五-----4
王五-----5 yeild一下
張三-----6
張三-----7
張三-----8
張三-----9
張三-----10 yeild一下
李四-----6
李四-----7
李四-----8
李四-----9
李四-----10 yeild一下
王五-----6
王五-----7
王五-----8
王五-----9
王五-----10 yeild一下
張三-----11
張三-----12
張三-----13
張三-----14
張三-----15 yeild一下
李四-----11
李四-----12
李四-----13
李四-----14
李四-----15 yeild一下
王五-----11
王五-----12
王五-----13
王五-----14
王五-----15 yeild一下
張三-----16
張三-----17
張三-----18
張三-----19
張三-----20 yeild一下
李四-----16
李四-----17
//其余的省略
款咖。何暮。。
yield原理
發(fā)現(xiàn)這一次铐殃,每次調(diào)用了yeild之后海洼,它們都沒有緊接著執(zhí)行了。這是為什么呢富腊,難道yield沒有用處了嗎坏逢?
其實上面出現(xiàn)的不確定結(jié)果,是因為多核CPU執(zhí)行的關(guān)系赘被,一個線程調(diào)用了yield方法之后是整,確實會讓出當前使用的CPU,讓自己從【運行態(tài)】變?yōu)椤揪途w態(tài)】民假。
- 當運行環(huán)境是單核CPU的時候浮入。如果其他線程已經(jīng)處于就緒態(tài),正在等待CPU時間片時羊异,這時有線程yield讓出了CPU時間片事秀,它們中的一個就會先有可能分配到CPU時間片彤断,進而進入【運行態(tài)】,執(zhí)行線程內(nèi)容易迹。
- 而當運行環(huán)境是多核CPU的話宰衙。也許上面的三個線程都是同時處于【運行態(tài)】正在執(zhí)行,那個一個線程yiedl之后睹欲,短暫的讓出了它的CPU供炼,而此時又沒有線程跟它搶CPU(因為其他兩個都在運行著),所以它可能又獲得了CPU時間片又去執(zhí)行了句伶。所以針對多核CPU環(huán)境的話劲蜻,測試結(jié)果并沒有明顯的規(guī)律。
yield錯誤觀點糾正
網(wǎng)上有說“yield之后考余,只有同優(yōu)先級的線程能執(zhí)行先嬉,低優(yōu)先級的線程無法獲得執(zhí)行”,這是錯誤的觀點楚堤,Java文檔并沒有這種說疫蔓,Java虛擬機也并沒有這種限制。會大概率出現(xiàn)這種情況主要是因為yield之后身冬,高優(yōu)先級的線程會更容易得到調(diào)度優(yōu)先得到CPU時間片執(zhí)行衅胀,低優(yōu)先級自然被執(zhí)行的概率就更少了,但是并不是說低優(yōu)先級線程就無法被執(zhí)行酥筝」銮可以把上面代碼的優(yōu)先級設(shè)置部分注釋去掉,自己親自運行看看結(jié)果嘿歌,別被誤導(dǎo)了掸掏。
yield的使用場景
yield的作用就是暫時讓出使用著的CPU,這樣其他【就緒態(tài)】的線程就有機會占用這個CPU去執(zhí)行宙帝。所以yield的使用場景多在丧凤,當前線程在進行耗時性的操作時(如IO操作),并且因為它的優(yōu)先級較高步脓,導(dǎo)致一些優(yōu)先級較低的線程被分配的時間片更少愿待,這樣優(yōu)先級低的線程就要等待更長時間才能完成操作,那么這時適當?shù)卣{(diào)用幾次yield方法讓出CPU靴患,讓優(yōu)先級低的線程多得到執(zhí)行仍侥,這樣才能高效的實現(xiàn)程序執(zhí)行和響應(yīng)。
測試例子查看 我的GitHub--JavaTest