Java線程Thread.yeild方法解析
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)歧義亚脆,我的理解是
造成當(dāng)前正在執(zhí)行的線程對象臨時性的暫停,和允許其他線程去執(zhí)行
也就是說山憨,調(diào)用了yield方法之后郁竟,當(dāng)前線程會臨時性的暫停一下,然后其他線程有機(jī)會去執(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++) {
// 當(dāng)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ù)字视哑,每當(dāng)打印的數(shù)字是5的倍數(shù)時(比如打印5,10,15等)挡毅,就yield一下。那么我們看看運(yùn)行之后的結(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í)行了奴紧。我們再運(yùn)行一次看看
張三-----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方法之后恩沽,確實會讓出當(dāng)前使用的CPU翔始,讓自己從【運(yùn)行態(tài)】變?yōu)椤揪途w態(tài)】罗心。
當(dāng)運(yùn)行環(huán)境是單核CPU的時候。如果其他線程已經(jīng)處于就緒態(tài)城瞎,正在等待CPU時間片時渤闷,這時有線程yield讓出了CPU時間片,它們中的一個就會先有可能分配到CPU時間片脖镀,進(jìn)而進(jìn)入【運(yùn)行態(tài)】飒箭,執(zhí)行線程內(nèi)容。
而當(dāng)運(yùn)行環(huán)境是多核CPU的話。也許上面的三個線程都是同時處于【運(yùn)行態(tài)】正在執(zhí)行弦蹂,那個一個線程yield之后肩碟,短暫的讓出了它的CPU,而此時又沒有線程跟它搶CPU(因為其他兩個都在運(yùn)行著)凸椿,所以它可能又獲得了CPU時間片又去執(zhí)行了。所以針對多核CPU環(huán)境的話削饵,測試結(jié)果并沒有明顯的規(guī)律岩瘦。
yield錯誤觀點(diǎn)糾正
網(wǎng)上有說“yield之后,只有同優(yōu)先級的線程能執(zhí)行窿撬,低優(yōu)先級的線程無法獲得執(zhí)行”启昧,這是錯誤的觀點(diǎn),Java文檔并沒有這種說劈伴,Java虛擬機(jī)也并沒有這種限制密末。會大概率出現(xiàn)這種情況主要是因為yield之后,高優(yōu)先級的線程會更容易得到調(diào)度優(yōu)先得到CPU時間片執(zhí)行跛璧,低優(yōu)先級自然被執(zhí)行的概率就更少了严里,但是并不是說低優(yōu)先級線程就無法被執(zhí)行∽烦牵可以把上面代碼的優(yōu)先級設(shè)置部分注釋去掉刹碾,自己親自運(yùn)行看看結(jié)果,別被誤導(dǎo)了座柱。
迷帜!拓展:Linux CPU的線程調(diào)度算法是:階梯算法,低優(yōu)先級的不一定執(zhí)行不到色洞,高優(yōu)先級的不一定一直執(zhí)行戏锹。
yield的使用場景
yield的作用就是暫時讓出使用著的CPU,這樣其他【就緒態(tài)】的線程就有機(jī)會占用這個CPU去執(zhí)行火诸。所以yield的使用場景多在锦针,當(dāng)前線程在進(jìn)行耗時性的操作時(如IO操作),并且因為它的優(yōu)先級較高置蜀,導(dǎo)致一些優(yōu)先級較低的線程被分配的時間片更少奈搜,這樣優(yōu)先級低的線程就要等待更長時間才能完成操作,那么這時適當(dāng)?shù)卣{(diào)用幾次yield方法讓出CPU盯荤,讓優(yōu)先級低的線程多得到執(zhí)行媚污,這樣才能高效的實現(xiàn)程序執(zhí)行和響應(yīng)。
測試?yán)硬榭?GitHub--JavaTest
大贊走一波廷雅,來簡書關(guān)注我