成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Thread.sleep(0)的意義& 多線程詳解

系統(tǒng)
我們可能經(jīng)常會用到 Thread.Sleep 函數(shù)來使線程掛起一段時間。那么你有沒有正確的理解這個函數(shù)的用法呢?

我們可能經(jīng)常會用到 Thread.Sleep 函數(shù)來使線程掛起一段時間。那么你有沒有正確的理解這個函數(shù)的用法呢?思考下面這兩個問題:

  • 假設現(xiàn)在是 2008-4-7 12:00:00.000,如果我調(diào)用一下 Thread.Sleep(1000) ,在 2008-4-7 12:00:01.000 的時候,這個線程會 不會被喚醒?
  • 某人的代碼中用了一句看似莫明其妙的話:Thread.Sleep(0) 。既然是 Sleep 0 毫秒,那么他跟去掉這句代碼相比,有啥區(qū)別么?

我們先回顧一下操作系統(tǒng)原理。

操作系統(tǒng)中,CPU競爭有很多種策略。Unix系統(tǒng)使用的是時間片算法,而Windows則屬于搶占式的。

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

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

在搶占式操作系統(tǒng)中,假設有若干進程,操作系統(tǒng)會根據(jù)他們的優(yōu)先級、饑餓時間(已經(jīng)多長時間沒有使用過 CPU 了),給他們算出一 個總的優(yōu)先級來。操作系統(tǒng)就會把 CPU 交給總優(yōu)先級最高的這個進程。當進程執(zhí)行完畢或者自己主動掛起后,操作系統(tǒng)就會重新計算一 次所有進程的總優(yōu)先級,然后再挑一個優(yōu)先級最高的把 CPU 控制權交給他。

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

如果是 Unix操作系統(tǒng)來負責分蛋糕,那么他會這樣定規(guī)矩:每個人上來吃 1 分鐘,時間到了換下一個。最后一個人吃完了就再從頭開始。于是,不管這10個人是不是優(yōu)先級不同、饑餓程度不同、飯量不同,每個人上來的時候都可以吃 1 分鐘。當然,如果有人本來不太餓,或者飯量小,吃了30秒鐘之后就吃飽了,那么他可以跟操作系統(tǒng)說:我已經(jīng)吃飽了(掛起)。于是操作系統(tǒng)就會讓下一個人接著來。

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

這樣看來,這個場面就有意思了——可能有些人是PPMM,因此具有高優(yōu)先級,于是她就可以經(jīng)常來吃蛋糕。可能另外一個人是個丑男,而去很ws,所以優(yōu)先級特別低,于是好半天了才輪到他一次(因為隨著時間的推移,他會越來越饑餓,因此算出來的總優(yōu)先級就會越來越高,因此總有一天會輪到他的)。而且,如果一不小心讓一個大胖子得到了刀叉,因為他飯量大,可能他會霸占著蛋糕連續(xù)吃很久很久,導致旁邊的人在那里咽口水。。。

而且,還可能會有這種情況出現(xiàn):操作系統(tǒng)現(xiàn)在計算出來的結果,5號PPMM總優(yōu)先級最高,而且高出別人一大截。因此就叫5號來吃蛋糕。5號吃了一小會兒,覺得沒那么餓了,于是說“我不吃了”(掛起)。因此操作系統(tǒng)就會重新計算所有人的優(yōu)先級。因為5號剛剛吃過,因此她的饑餓程度變小了,于是總優(yōu)先級變小了;而其他人因為多等了一會兒,饑餓程度都變大了,所以總優(yōu)先級也變大了。不過這時候仍然有可能5號的優(yōu)先級比別的都高,只不過現(xiàn)在只比其他的高一點點——但她仍然是總優(yōu)先級最高的啊。因此操作系統(tǒng)就會說:5號mm上來吃蛋糕……(5號mm心里郁悶,這不剛吃過嘛……人家要減肥……誰叫你長那么漂亮,獲得了那么高的優(yōu)先級)。

那么,Thread.Sleep 函數(shù)是干嗎的呢?還用剛才的分蛋糕的場景來描述。上面的場景里面,5號MM在吃了一次蛋糕之后,覺得已經(jīng)有8分飽了,她覺得在未來的半個小時之內(nèi)都不想再來吃蛋糕了,那么她就會跟操作系統(tǒng)說:在未來的半個小時之內(nèi)不要再叫我上來吃蛋糕了。這樣,操作系統(tǒng)在隨后的半個小時里面重新計算所有人總優(yōu)先級的時候,就會忽略5號mm。Sleep函數(shù)就是干這事的,他告訴操作系統(tǒng)“在未來的多少毫秒內(nèi)我不參與CPU競爭”。

看完了 Thread.Sleep 的作用,我們再來想想文章開頭的兩個問題。

對于第一個問題,答案是:不一定。因為你只是告訴操作系統(tǒng):在未來的1000毫秒內(nèi)我不想再參與到CPU競爭。那么1000毫秒過去之后,這時候也許另外一個線程正在使用CPU,那么這時候操作系統(tǒng)是不會重新分配CPU的,直到那個線程掛起或結束;況且,即使這個時候恰巧輪到操作系統(tǒng)進行CPU 分配,那么當前線程也不一定就是總優(yōu)先級最高的那個,CPU還是可能被其他線程搶占去。

與此相似的,Thread有個Resume函數(shù),是用來喚醒掛起的線程的。好像上面所說的一樣,這個函數(shù)只是“告訴操作系統(tǒng)我從現(xiàn)在起開始參與CPU競爭了”,這個函數(shù)的調(diào)用并不能馬上使得這個線程獲得CPU控制權。

對于第二個問題,答案是:有,而且區(qū)別很明顯。假設我們剛才的分蛋糕場景里面,有另外一個PPMM 7號,她的優(yōu)先級也非常非常高(因為非常非常漂亮),所以操作系統(tǒng)總是會叫道她來吃蛋糕。而且,7號也非常喜歡吃蛋糕,而且飯量也很大。不過,7號人品很好,她很善良,她沒吃幾口就會想:如果現(xiàn)在有別人比我更需要吃蛋糕,那么我就讓給他。因此,她可以每吃幾口就跟操作系統(tǒng)說:我們來重新計算一下所有人的總優(yōu)先級吧。不過,操作系統(tǒng)不接受這個建議——因為操作系統(tǒng)不提供這個接口。于是7號mm就換了個說法:“在未來的0毫秒之內(nèi)不要再叫我上來吃蛋糕了”。這個指令操作系統(tǒng)是接受的,于是此時操作系統(tǒng)就會重新計算大家的總優(yōu)先級——注意這個時候是連7號一起計算的,因為“0毫秒已經(jīng)過去了”嘛。因此如果沒有比7號更需要吃蛋糕的人出現(xiàn),那么下一次7號還是會被叫上來吃蛋糕。

因此,Thread.Sleep(0)的作用,就是“觸發(fā)操作系統(tǒng)立刻重新進行一次CPU競爭”。競爭的結果也許是當前線程仍然獲得CPU控制權,也許會換成別的線程獲得CPU控制權。這也是我們在大循環(huán)里面經(jīng)常會寫一句Thread.Sleep(0) ,因為這樣就給了其他線程比如Paint線程獲得CPU控制權的權力,這樣界面就不會假死在那里。

另外,雖然上面提到說“除非它自己放棄使用 CPU ,否則將完全霸占 CPU”,但這個行為仍然是受到制約的——操作系統(tǒng)會監(jiān)控你霸占CPU的情況,如果發(fā)現(xiàn)某個線程長時間霸占CPU,會強制使這個線程掛起,因此在實際上不會出現(xiàn)“一個線程一直霸占著 CPU 不放”的情況。至于我們的大循環(huán)造成程序假死,并不是因為這個線程一直在霸占著CPU。實際上在這段時間操作系統(tǒng)已經(jīng)進行過多次CPU競爭了,只不過其他線程在獲得CPU控制權之后很短時間內(nèi)馬上就退出了,于是就又輪到了這個線程繼續(xù)執(zhí)行循環(huán),于是就又用了很久才被操作系統(tǒng)強制掛起。。。因此反應到界面上,看起來就好像這個線程一直在霸占著CPU一樣。

末了再說明一下,文中線程、進程有點混亂,其實在Windows原理層面,CPU競爭都是線程級的,本文中把這里的進程、線程看成同一個東西就好了。?

責任編輯:趙寧寧 來源: 博客園
相關推薦

2024-11-18 17:06:11

Java線程

2022-12-28 08:10:31

ThreadjavaGC

2022-05-24 12:57:49

函數(shù)代碼Java

2023-05-10 07:42:26

Java多線程編程

2022-04-18 07:36:37

TimeUnit線程休眠

2009-06-29 17:54:10

Java多線程Thread類創(chuàng)建線程

2024-04-12 09:02:15

JavaCPU執(zhí)行時間線程

2009-10-12 13:32:58

VB.NET線程構造器

2021-03-09 08:03:21

Node.js 線程JavaScript

2022-05-20 10:20:17

Spring事務MyBatis

2010-04-14 09:20:26

.NET多線程

2010-03-17 14:58:20

Java多線程

2021-12-28 09:10:55

Java線程狀態(tài)

2024-10-24 17:13:55

WinformUI多線程

2009-03-24 08:56:23

數(shù)據(jù)同步多線程Java

2023-10-18 15:19:56

2009-03-26 22:54:33

IBM院士Sun

2011-06-22 15:09:34

Qt 線程 sleep

2011-08-31 16:30:19

Lua多線程

2024-12-23 09:09:54

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久av资源网| 91精品免费 | 国产网站在线播放 | 国产精品视频yy9299一区 | 99精品国产一区二区青青牛奶 | 亚洲天堂av网 | 精品国产31久久久久久 | 小h片免费观看久久久久 | 亚洲国产成人精品女人久久久野战 | 亚洲一区二区中文字幕 | 久久久久国产成人精品亚洲午夜 | 美日韩精品| 懂色一区二区三区免费观看 | 亚洲精品一区二区另类图片 | 2019中文字幕视频 | 国产伦一区二区三区视频 | 奇米久久 | 久久久久国产精品 | 中国一级大黄大片 | 亚洲成人免费视频 | 亚洲午夜av久久乱码 | 亚洲精彩视频在线观看 | 中文字幕亚洲精品 | 夜夜艹天天干 | 激情在线视频 | 国产乱人伦精品一区二区 | 国产精品亚洲欧美日韩一区在线 | 中文字幕国产第一页 | 99re99| 精品久久久网站 | 精品三级在线观看 | 欧美一区二区三区在线 | 免费激情网站 | 国外成人在线视频 | 亚洲一区视频在线 | 秋霞av国产精品一区 | 国产精品不卡视频 | 东京av男人的天堂 | 中文字幕日韩欧美 | 啪啪av| 成人精品国产免费网站 |