閑魚一面:Thread.sleep(0) 到底有什麼用?

来源:https://www.cnblogs.com/javastack/archive/2022/05/24/16305196.html
-Advertisement-
Play Games

來源:www.cnblogs.com/keyyang/p/4128424.html 我們可能經常會用到 Thread.Sleep 函數來使線程掛起一段時間。那麼你有沒有正確的理解這個函數的用法呢?思考下麵這兩個問題: 假設現在是 2008-4-7 12:00:00.000,如果我調用一下 Threa ...


來源:www.cnblogs.com/keyyang/p/4128424.html

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

假設現在是 2008-4-7 12:00:00.000,如果我調用一下 Thread.Sleep(1000) ,在 2008-4-7 12:00:01.000 的時候,這個線程會 不會被喚醒?

某人的代碼中用了一句看似莫明其妙的話:Thread.Sleep(0) 。既然是 Sleep 0 毫秒,那麼他跟去掉這句代碼相比,有啥區別麽?
我們先回顧一下操作系統原理。

Java 多線程系統教程:https://www.javastack.cn/categories/Java/多線程/

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

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

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

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

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

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

如果是 Windows 操作系統來負責分蛋糕的,那麼場面就很有意思了。他會這樣定規矩:我會根據你們的優先順序、饑餓程度去給你們每個人計算一個優先順序。優先順序最高的那個人,可以上來吃蛋糕——吃到你不想吃為止。等這個人吃完了,我再重新根據優先順序、饑餓程度來計算每個人的優先順序,然後再分給優先順序最高的那個人。

這樣看來,這個場面就有意思了——可能有些人是PPMM,因此具有高優先順序,於是她就可以經常來吃蛋糕。可能另外一個人是個醜男,而去很ws,所以優先順序特別低,於是好半天了才輪到他一次(因為隨著時間的推移,他會越來越饑餓,因此算出來的總優先順序就會越來越高,因此總有一天會輪到他的)。

而且,如果一不小心讓一個大胖子得到了刀叉,因為他飯量大,可能他會霸占著蛋糕連續吃很久很久,導致旁邊的人在那裡咽口水。。。

而且,還可能會有這種情況出現:操作系統現在計算出來的結果,5號PPMM總優先順序最高,而且高出別人一大截。因此就叫5號來吃蛋糕。5號吃了一小會兒,覺得沒那麼餓了,於是說“我不吃了”(掛起)。

因此操作系統就會重新計算所有人的優先順序。因為5號剛剛吃過,因此她的饑餓程度變小了,於是總優先順序變小了;而其他人因為多等了一會兒,饑餓程度都變大了,所以總優先順序也變大了。不過這時候仍然有可能5號的優先順序比別的都高,只不過現在只比其他的高一點點——但她仍然是總優先順序最高的啊。因此操作系統就會說:5號mm上來吃蛋糕……(5號mm心裡鬱悶,這不剛吃過嘛……人家要減肥……誰叫你長那麼漂亮,獲得了那麼高的優先順序)。

那麼,Thread.Sleep 函數是幹嗎的呢?還用剛纔的分蛋糕的場景來描述。上面的場景裡面,5號MM在吃了一次蛋糕之後,覺得已經有8分飽了,她覺得在未來的半個小時之內都不想再來吃蛋糕了,那麼她就會跟操作系統說:在未來的半個小時之內不要再叫我上來吃蛋糕了。這樣,操作系統在隨後的半個小時裡面重新計算所有人總優先順序的時候,就會忽略5號mm。Sleep函數就是乾這事的,他告訴操作系統“在未來的多少毫秒內我不參與CPU競爭”。

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

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

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

對於第二個問題,答案是:有,而且區別很明顯。假設我們剛纔的分蛋糕場景裡面,有另外一個PPMM 7號,她的優先順序也非常非常高(因為非常非常漂亮),所以操作系統總是會叫道她來吃蛋糕。

而且,7號也非常喜歡吃蛋糕,而且飯量也很大。不過,7號人品很好,她很善良,她沒吃幾口就會想:如果現在有別人比我更需要吃蛋糕,那麼我就讓給他。因此,她可以每吃幾口就跟操作系統說:我們來重新計算一下所有人的總優先順序吧。不過,操作系統不接受這個建議——因為操作系統不提供這個介面。

於是7號mm就換了個說法:“在未來的0毫秒之內不要再叫我上來吃蛋糕了”。這個指令操作系統是接受的,於是此時操作系統就會重新計算大家的總優先順序——註意這個時候是連7號一起計算的,因為“0毫秒已經過去了”嘛。因此如果沒有比7號更需要吃蛋糕的人出現,那麼下一次7號還是會被叫上來吃蛋糕。

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

另外,雖然上面提到說“除非它自己放棄使用 CPU ,否則將完全霸占 CPU”,但這個行為仍然是受到制約的——操作系統會監控你霸占CPU的情況,如果發現某個線程長時間霸占CPU,會強制使這個線程掛起,因此在實際上不會出現“一個線程一直霸占著 CPU 不放”的情況。

至於我們的大迴圈造成程式假死,並不是因為這個線程一直在霸占著CPU。實際上在這段時間操作系統已經進行過多次CPU競爭了,只不過其他線程在獲得CPU控制權之後很短時間內馬上就退出了,於是就又輪到了這個線程繼續執行迴圈,於是就又用了很久才被操作系統強制掛起。。。因此反應到界面上,看起來就好像這個線程一直在霸占著CPU一樣。

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

近期熱文推薦:

1.1,000+ 道 Java面試題及答案整理(2022最新版)

2.勁爆!Java 協程要來了。。。

3.Spring Boot 2.x 教程,太全了!

4.別再寫滿屏的爆爆爆炸類了,試試裝飾器模式,這才是優雅的方式!!

5.《Java開發手冊(嵩山版)》最新發佈,速速下載!

覺得不錯,別忘了隨手點贊+轉發哦!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在kubernetes容器環境下 kafka會預設把主機名註冊到zookeeper。這個時候消費端部署在不同的命名空間或者不同的集群中會出現無法訪問的情況。用advertised.listeners配置可以重寫預設註冊的地址。 定義 listeners listeners 配置的是kafka Ser ...
  • 到底是什麼面試題, 讓一個工作了4年的精神小伙,只是去參加了一場技術面試, 就被搞得精神萎靡。鬱郁寡歡! 這一切的背後到底是道德的淪喪,還是人性的扭曲。 讓我們一起揭秘一下這道面試題。 關於, “簡述你對線程池的理解”,看看普通人和高手的回答。 普通人: 嗯。。。。。。。。。。 高手: 關於這個問題 ...
  • 目錄 一.簡介 二.效果演示 三.源碼下載 四.猜你喜歡 零基礎 OpenGL (ES) 學習路線推薦 : OpenGL (ES) 學習目錄 >> OpenGL ES 基礎 零基礎 OpenGL (ES) 學習路線推薦 : OpenGL (ES) 學習目錄 >> OpenGL ES 轉場 零基礎 O ...
  • 每日一句 軍人天生就捨棄了戰鬥的意義! 概述 RabitMQ 發佈確認,保證消息在磁碟上。 前提條件 1。隊列必須持久化 隊列持久化 2。隊列中的消息必須持久化 消息持久化 使用 三種發佈確認的方式: 1。單個發佈確認 2。批量發佈確認 3。非同步批量發佈確認 開啟發佈確認的方法 //創建一個連接工廠 ...
  • 使用FFmpeg庫做的項目,調試項目的時候發現,連續解視頻進行播放,會有明顯記憶體增加問題。連續工作10個小時後就會 被linux 內核kill掉。 通過逐步註掉代碼和網上查閱資料。最後發現記憶體泄漏有一些幾個地方: 一、av_read_frame的問題 從網上查閱大神們的經驗,主要是av_read_f ...
  • 在Java 9中又新增了一些API來幫助便捷的創建不可變集合,以減少代碼複雜度。 本期配套視頻:Java 9 新特性:快速定義不可變集合 常規寫法 以往我們創建一些不可變集合的時候,通常是這樣寫的: // 不可變的Set Set<String> set = new HashSet<>(); set. ...
  • 反向代理(2022/03/31) 簡單記錄 Nginx 反向代理相關的一些配置文件,描述不足之處請自行查閱相關資料。 1. HTTP 配置 upstream web { server domain.com:80; } server { # 監聽 tcp4 listen 80; # 監聽 tcp6 l ...
  • 爬蟲代理 IP 池及隧道代理 日常開發中,偶爾會遇到爬取網頁數據的需求,為了隱藏本機真實 IP,常常會用到代理 IP 池,本文將基於 openresty 與代理 IP 池搭建更為易用的隧道代理。 1. 代理 IP 池 1.1 簡介 代理 IP 池即在資料庫中維護一個可用的 IP 代理隊列,一般實現思 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...