欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談Thread.sleep(0)到底有什么用

 更新時(shí)間:2022年06月01日 10:48:21   作者:ZNineSun  
為什么要用sleep,主要是為了暫停當(dāng)前線程,把cpu片段讓出給其他線程,減緩當(dāng)前線程的執(zhí)行,本文主要介紹了Thread.sleep(0)到底有什么用,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

我們可能經(jīng)常會(huì)用到Thead.sleep()函數(shù)來吧使線程掛起一段時(shí)間。但是你真的了解這個(gè)函數(shù)的真正作用嗎?

先思考兩個(gè)問題:

假設(shè)現(xiàn)在是 2022-5-26 12:00:00.000,如果我調(diào)用一下 Thread.Sleep(1000) ,在 2022-5-26 12:00:01.000 的時(shí)候,這個(gè)線程會(huì)不會(huì)被喚醒?

Thread.Sleep(0) 。既然是 Sleep 0 毫秒,那么他跟去掉這句代碼相比,有啥區(qū)別么?

其實(shí)回答這個(gè)問題的本質(zhì)就是操作系統(tǒng)對(duì)資源的分配

不管有沒有學(xué)習(xí)過操作系統(tǒng)的小伙伴,都一起看一下操作系統(tǒng)的原理。

操作系統(tǒng)中,CPU競(jìng)爭(zhēng)有很多種策略:

  • Unix系統(tǒng)使用的是時(shí)間片算法
  • 而Windows則屬于搶占式的。

在時(shí)間片算法中,所有的進(jìn)程排成一個(gè)隊(duì)列。操作系統(tǒng)按照他們的順序,給每個(gè)進(jìn)程分配一段時(shí)間,即該進(jìn)程允許運(yùn)行的時(shí)間。如果在時(shí)間片結(jié)束時(shí)進(jìn)程還在運(yùn)行,則CPU將被剝奪并分配給另一個(gè)進(jìn)程。如果進(jìn)程在時(shí)間片結(jié)束前阻塞或結(jié)束,則CPU當(dāng)即進(jìn)行切換。調(diào)度程序所要做的就是維護(hù)一張就緒進(jìn)程列表,當(dāng)進(jìn)程用完它的時(shí)間片后,它被移到隊(duì)列的末尾。

所謂搶占式操作系統(tǒng),就是說如果一個(gè)進(jìn)程得到了 CPU 時(shí)間,除非它自己放棄使用 CPU ,否則將完全霸占 CPU 。因此可以看出,在搶占式操作系統(tǒng)中,操作系統(tǒng)假設(shè)所有的進(jìn)程都是“人品很好”的,會(huì)主動(dòng)退出 CPU 。

在搶占式操作系統(tǒng)中,假設(shè)有若干進(jìn)程,操作系統(tǒng)會(huì)根據(jù)他們的優(yōu)先級(jí)、饑餓時(shí)間(已經(jīng)多長(zhǎng)時(shí)間沒有使用過 CPU 了),給他們算出一個(gè)總的優(yōu)先級(jí)來。操作系統(tǒng)就會(huì)把 CPU 交給總優(yōu)先級(jí)最高的這個(gè)進(jìn)程。

當(dāng)進(jìn)程執(zhí)行完畢或者自己主動(dòng)掛起后,操作系統(tǒng)就會(huì)重新計(jì)算一 次所有進(jìn)程的總優(yōu)先級(jí),然后再挑一個(gè)優(yōu)先級(jí)最高的把 CPU 控制權(quán)交給他。

我們用分蛋糕的場(chǎng)景來描述這兩種算法。假設(shè)有源源不斷的蛋糕(源源不斷的時(shí)間),一副刀叉(一個(gè)CPU),10個(gè)等待吃蛋糕的人(10 個(gè)進(jìn)程)。

先來看看unix系統(tǒng)是怎么分蛋糕的:

如果是 Unix操作系統(tǒng)來負(fù)責(zé)分蛋糕,那么他會(huì)這樣定規(guī)矩:每個(gè)人上來吃 1 分鐘,時(shí)間到了換下一個(gè)。最后一個(gè)人吃完了就再從頭開始。于是,不管這10個(gè)人是不是優(yōu)先級(jí)不同、饑餓程度不同、飯量不同,每個(gè)人上來的時(shí)候都可以吃 1 分鐘。

當(dāng)然,如果有人本來不太餓,或者飯量小,吃了30秒鐘之后就吃飽了,那么他可以跟操作系統(tǒng)說:我已經(jīng)吃飽了(掛起)。于是操作系統(tǒng)就會(huì)讓下一個(gè)人接著來。

如果是 Windows 操作系統(tǒng)來負(fù)責(zé)分蛋糕的,那么場(chǎng)面就很有意思了。他會(huì)這樣定規(guī)矩:我會(huì)根據(jù)你們的優(yōu)先級(jí)、饑餓程度去給你們每個(gè)人計(jì)算一個(gè)優(yōu)先級(jí)。優(yōu)先級(jí)最高的那個(gè)人,可以上來吃蛋糕——吃到你不想吃為止。等這個(gè)人吃完了,我再重新根據(jù)優(yōu)先級(jí)、饑餓程度來計(jì)算每個(gè)人的優(yōu)先級(jí),然后再分給優(yōu)先級(jí)最高的那個(gè)人。

這樣看來,這個(gè)場(chǎng)面就有意思了:可能有些人是個(gè)帥鍋或者是個(gè)美眉,因此具有高優(yōu)先級(jí),于是她就可以經(jīng)常來吃蛋糕??赡芰硗庖粋€(gè)人是個(gè)丑男,而去很ws,所以優(yōu)先級(jí)特別低,于是好半天了才輪到他一次(因?yàn)殡S著時(shí)間的推移,他會(huì)越來越饑餓,因此算出來的總優(yōu)先級(jí)就會(huì)越來越高,因此總有一天會(huì)輪到他的)。

而且,如果一不小心讓一個(gè)大胖子得到了刀叉,因?yàn)樗埩看?,可能他?huì)霸占著蛋糕連續(xù)吃很久很久,導(dǎo)致旁邊的人在那里咽口水~

而且,還可能會(huì)有這種情況出現(xiàn):操作系統(tǒng)現(xiàn)在計(jì)算出來的結(jié)果,5號(hào)PPMM總優(yōu)先級(jí)最高,而且高出別人一大截。因此就叫5號(hào)來吃蛋糕。5號(hào)吃了一小會(huì)兒,覺得沒那么餓了,于是說“我不吃了”(掛起)。因此操作系統(tǒng)就會(huì)重新計(jì)算所有人的優(yōu)先級(jí)。

因?yàn)?號(hào)剛剛吃過,因此她的饑餓程度變小了,于是總優(yōu)先級(jí)變小了;而其他人因?yàn)槎嗟攘艘粫?huì)兒,饑餓程度都變大了,所以總優(yōu)先級(jí)也變大了。不過這時(shí)候仍然有可能5號(hào)的優(yōu)先級(jí)比別的都高,只不過現(xiàn)在只比其他的高一點(diǎn)點(diǎn)——但她仍然是總優(yōu)先級(jí)最高的啊。
因此操作系統(tǒng)就會(huì)說:5號(hào)mm上來吃蛋糕……(5號(hào)mm心里郁悶,這不剛吃過嘛……人家要減肥……誰叫你長(zhǎng)那么漂亮,獲得了那么高的優(yōu)先級(jí))。

那么問題來了,說了這一堆,Thread.Sleep 函數(shù)到底是干嗎的呢?

還用剛才的分蛋糕的場(chǎng)景來描述。上面的場(chǎng)景里面,5號(hào)MM在吃了一次蛋糕之后,覺得已經(jīng)有8分飽了,她覺得在未來的半個(gè)小時(shí)之內(nèi)都不想再來吃蛋糕了,那么她就會(huì)跟操作系統(tǒng)說:在未來的半個(gè)小時(shí)之內(nèi)不要再叫我上來吃蛋糕了。

這樣,操作系統(tǒng)在隨后的半個(gè)小時(shí)里面重新計(jì)算所有人總優(yōu)先級(jí)的時(shí)候,就會(huì)忽略5號(hào)mm。Sleep函數(shù)就是干這事的,他告訴操作系統(tǒng)“在未來的多少毫秒內(nèi)我不參與CPU競(jìng)爭(zhēng)”。

知道了Sleep函數(shù)的作用之后,在回過頭來繼續(xù)看看文章初始時(shí)提到的兩個(gè)問題:

1.假設(shè)現(xiàn)在是 2022-5-26 12:00:00.000,如果我調(diào)用一下 Thread.Sleep(1000) ,在 2022-5-26 12:00:01.000 的時(shí)候,這個(gè)線程會(huì)不會(huì)被喚醒?

答案是:不一定。因?yàn)槟阒皇歉嬖V操作系統(tǒng):在未來的1000毫秒內(nèi)我不想再參與到CPU競(jìng)爭(zhēng)。那么1000毫秒過去之后,這時(shí)候也許另外一個(gè)線程正在使用CPU,那么這時(shí)候操作系統(tǒng)是不會(huì)重新分配CPU的,直到那個(gè)線程掛起或結(jié)束;況且,即使這個(gè)時(shí)候恰巧輪到操作系統(tǒng)進(jìn)行CPU 分配,那么當(dāng)前線程也不一定就是總優(yōu)先級(jí)最高的那個(gè),CPU還是可能被其他線程搶占去。
與此相似的,Thread有個(gè)Resume函數(shù),是用來喚醒掛起的線程的。好像上面所說的一樣,這個(gè)函數(shù)只是“告訴操作系統(tǒng)我從現(xiàn)在起開始參與CPU競(jìng)爭(zhēng)了”,這個(gè)函數(shù)的調(diào)用并不能馬上使得這個(gè)線程獲得CPU控制權(quán)。

2.使用Thread.Sleep(0)跟去掉這句代碼相比,有啥區(qū)別么?

答案是:有,而且區(qū)別很明顯。假設(shè)我們剛才的分蛋糕場(chǎng)景里面,有另外一個(gè)PPMM 7號(hào),她的優(yōu)先級(jí)也非常非常高(因?yàn)榉浅7浅F粒?,所以操作系統(tǒng)總是會(huì)叫道她來吃蛋糕。而且,7號(hào)也非常喜歡吃蛋糕,而且飯量也很大。不過,7號(hào)人品很好,她很善良,她沒吃幾口就會(huì)想:如果現(xiàn)在有別人比我更需要吃蛋糕,那么我就讓給他。

因此,她可以每吃幾口就跟操作系統(tǒng)說:我們來重新計(jì)算一下所有人的總優(yōu)先級(jí)吧。不過,操作系統(tǒng)不接受這個(gè)建議——因?yàn)椴僮飨到y(tǒng)不提供這個(gè)接口。于是7號(hào)mm就換了個(gè)說法:“在未來的0毫秒之內(nèi)不要再叫我上來吃蛋糕了”。這個(gè)指令操作系統(tǒng)是接受的,于是此時(shí)操作系統(tǒng)就會(huì)重新計(jì)算大家的總優(yōu)先級(jí)——注意這個(gè)時(shí)候是連7號(hào)一起計(jì)算的,因?yàn)?ldquo;0毫秒已經(jīng)過去了”嘛。因此如果沒有比7號(hào)更需要吃蛋糕的人出現(xiàn),那么下一次7號(hào)還是會(huì)被叫上來吃蛋糕。

因此,Thread.Sleep(0)的作用,就是“觸發(fā)操作系統(tǒng)立刻重新進(jìn)行一次CPU競(jìng)爭(zhēng)”。競(jìng)爭(zhēng)的結(jié)果也許是當(dāng)前線程仍然獲得CPU控制權(quán),也許會(huì)換成別的線程獲得CPU控制權(quán)。這也是我們?cè)诖笱h(huán)里面經(jīng)常會(huì)寫一句Thread.Sleep(0) ,因?yàn)檫@樣就給了其他線程比如Paint線程獲得CPU控制權(quán)的權(quán)力,這樣界面就不會(huì)假死在那里。

另外,雖然上面提到說“除非它自己放棄使用 CPU ,否則將完全霸占 CPU”,但這個(gè)行為仍然是受到制約的——操作系統(tǒng)會(huì)監(jiān)控你霸占CPU的情況,如果發(fā)現(xiàn)某個(gè)線程長(zhǎng)時(shí)間霸占CPU,會(huì)強(qiáng)制使這個(gè)線程掛起,因此在實(shí)際上不會(huì)出現(xiàn)“一個(gè)線程一直霸占著 CPU 不放”的情況。至于我們的大循環(huán)造成程序假死,并不是因?yàn)檫@個(gè)線程一直在霸占著CPU。

實(shí)際上在這段時(shí)間操作系統(tǒng)已經(jīng)進(jìn)行過多次CPU競(jìng)爭(zhēng)了,只不過其他線程在獲得CPU控制權(quán)之后很短時(shí)間內(nèi)馬上就退出了,于是就又輪到了這個(gè)線程繼續(xù)執(zhí)行循環(huán),于是就又用了很久才被操作系統(tǒng)強(qiáng)制掛起。。。因此反應(yīng)到界面上,看起來就好像這個(gè)線程一直在霸占著CPU一樣。

關(guān)于sleep()方法和yield()方法的區(qū)別如下

  • sleep()方法暫停 當(dāng)前線程后,會(huì)給其他線程執(zhí)行機(jī)會(huì),不會(huì)理會(huì)其他線程的優(yōu)先級(jí):但yield()方法只會(huì)給優(yōu)先級(jí)相同,或優(yōu)先級(jí)更高的線程執(zhí)行機(jī)會(huì)。
  • sleep()方法 會(huì)將線程轉(zhuǎn)入阻塞狀態(tài),直到經(jīng)過阻塞時(shí)間才會(huì)轉(zhuǎn)入就緒狀態(tài);而yield()不會(huì)將線程轉(zhuǎn)入阻塞狀態(tài),它只是強(qiáng)制當(dāng)前線程進(jìn)入就緒狀態(tài)。因此完全有可能某個(gè)線程調(diào)用yield()方法暫停之后,立即再次獲得處理器資源被執(zhí)行。
  • sleep()方法聲明拋出了InterruptedException異常,所以調(diào)用sleep()方法時(shí)要么捕捉該異常,要么顯式聲明拋出該異常:而yield()方法則沒有聲明拋出任何異常。
  • sleep()方法比yield()方法有更好的可移植性,通常不建議使用yield()方法來控制并發(fā)線程的執(zhí)行。

 到此這篇關(guān)于淺談Thread.sleep(0)到底有什么用的文章就介紹到這了,更多相關(guān)Thread.sleep(0)作用內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java中使用MinIO的常用操作示例

    Java中使用MinIO的常用操作示例

    這篇文章主要介紹了Java中MinIO的常用操作示例,MinIO 是一款基于Go語言發(fā)開的高性能、分布式的對(duì)象存儲(chǔ)系統(tǒng),客戶端支持Java,Net,Python,Javacript, Golang語言,需要的朋友可以參考下
    2024-01-01
  • Spring Boot 中的默認(rèn)異常處理機(jī)制解析(如 /error 接口)

    Spring Boot 中的默認(rèn)異常處理機(jī)制解析(如 /error 接口)

    SpringBoot通過/error接口默認(rèn)處理異常,使用BasicErrorController返回結(jié)構(gòu)化JSON或HTML錯(cuò)誤頁面,支持自定義ErrorAttributes、ControllerAdvice及錯(cuò)誤模板,配置項(xiàng)可調(diào)整錯(cuò)誤路徑和格式,本文給大家介紹Spring Boot 中的默認(rèn)異常處理機(jī)制解析,感興趣的朋友一起看看吧
    2025-07-07
  • 詳解Java的readBytes是怎么實(shí)現(xiàn)的

    詳解Java的readBytes是怎么實(shí)現(xiàn)的

    眾所周知,Java是一門跨平臺(tái)語言,針對(duì)不同的操作系統(tǒng)有不同的實(shí)現(xiàn),下面小編就來從一個(gè)非常簡(jiǎn)單的api調(diào)用帶大家來看看Java具體是怎么做的吧
    2023-07-07
  • 淺談Java編程ToString()方法重寫的意義

    淺談Java編程ToString()方法重寫的意義

    這篇文章主要介紹了淺談Java編程ToString()方法重寫的意義,還是挺不錯(cuò)的,這里分享給大家,供朋友們學(xué)習(xí)和參考。
    2017-10-10
  • Java中的方法、常量、變量、參數(shù)用例詳解

    Java中的方法、常量、變量、參數(shù)用例詳解

    在JVM的運(yùn)轉(zhuǎn)中,承載的是數(shù)據(jù),而數(shù)據(jù)的一種變現(xiàn)形式就是“量”,量分為:常量與變量,我們?cè)跀?shù)學(xué)和物理學(xué)中已經(jīng)接觸過變量的概念了,在Java中的變量就是在程序運(yùn)行過程中可以改變其值的量,這篇文章主要介紹了Java中的方法、常量、變量、參數(shù),需要的朋友可以參考下
    2024-01-01
  • Java開發(fā)中常用記錄

    Java開發(fā)中常用記錄

    這篇文章主要介紹了Java-編程式事務(wù)、Java-Stream、Linux常用命令,需要的朋友可以參考下
    2023-05-05
  • Java日常練習(xí)題,每天進(jìn)步一點(diǎn)點(diǎn)(55)

    Java日常練習(xí)題,每天進(jìn)步一點(diǎn)點(diǎn)(55)

    下面小編就為大家?guī)硪黄狫ava基礎(chǔ)的幾道練習(xí)題(分享)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧,希望可以幫到你
    2021-08-08
  • 五個(gè)很實(shí)用的IDEA使用技巧分享

    五個(gè)很實(shí)用的IDEA使用技巧分享

    IntelliJ IDEA 是一款優(yōu)秀的 Java 集成開發(fā)環(huán)境,它提供了許多強(qiáng)大的功能和快捷鍵,可以幫助開發(fā)者提高編碼效率和質(zhì)量,本文就在為你介紹博主常用的五個(gè)IntelliJ IDEA使用技巧,希望能夠給你帶來一些工作效率上的提升
    2023-10-10
  • Java虛擬機(jī)內(nèi)存區(qū)域劃分詳解

    Java虛擬機(jī)內(nèi)存區(qū)域劃分詳解

    這篇文章主要介紹了Java虛擬機(jī)內(nèi)存區(qū)域劃分,本文邏輯清晰,可以幫助我們更好的掌握虛擬機(jī),對(duì)我們學(xué)習(xí)java來說是一種幫助,需要的朋友可以參考下
    2021-04-04
  • SpringBoot之解決多個(gè)定時(shí)任務(wù)阻塞的問題

    SpringBoot之解決多個(gè)定時(shí)任務(wù)阻塞的問題

    這篇文章主要介紹了SpringBoot之解決多個(gè)定時(shí)任務(wù)阻塞的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-04-04

最新評(píng)論