国产精品久久久aaaa,日日干夜夜操天天插,亚洲乱熟女香蕉一区二区三区少妇,99精品国产高清一区二区三区,国产成人精品一区二区色戒,久久久国产精品成人免费,亚洲精品毛片久久久久,99久久婷婷国产综合精品电影,国产一区二区三区任你鲁

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

編程中常見的幾種鎖

數據分析與開發 ? 來源:數據分析與開發 ? 作者:數據分析與開發 ? 2020-10-30 16:27 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

前言

生活中用到的鎖,用途都比較簡單粗暴,上鎖基本是為了防止外人進來、電動車被偷等等。

但生活中也不是沒有 BUG 的,比如加鎖的電動車在「廣西 - 竊·格瓦拉」面前,鎖就是形同虛設,只要他愿意,他就可以輕輕松松地把你電動車給「順走」,不然打工怎么會是他這輩子不可能的事情呢?牛逼之人,必有牛逼之處。

那在編程世界里,「鎖」更是五花八門,多種多樣,每種鎖的加鎖開銷以及應用場景也可能會不同。

如何用好鎖,也是程序員的基本素養之一了。

高并發的場景下,如果選對了合適的鎖,則會大大提高系統的性能,否則性能會降低。

所以,知道各種鎖的開銷,以及應用場景是很有必要的。

接下來,就談一談常見的這幾種鎖:

正文

多線程訪問共享資源的時候,避免不了資源競爭而導致數據錯亂的問題,所以我們通常為了解決這一問題,都會在訪問共享資源之前加鎖。

最常用的就是互斥鎖,當然還有很多種不同的鎖,比如自旋鎖、讀寫鎖、樂觀鎖等,不同種類的鎖自然適用于不同的場景。

如果選擇了錯誤的鎖,那么在一些高并發的場景下,可能會降低系統的性能,這樣用戶體驗就會非常差了。

所以,為了選擇合適的鎖,我們不僅需要清楚知道加鎖的成本開銷有多大,還需要分析業務場景中訪問的共享資源的方式,再來還要考慮并發訪問共享資源時的沖突概率。

對癥下藥,才能減少鎖對高并發性能的影響。

那接下來,針對不同的應用場景,談一談「互斥鎖、自旋鎖、讀寫鎖、樂觀鎖、悲觀鎖」的選擇和使用。

互斥鎖與自旋鎖:誰更輕松自如?

最底層的兩種就是會「互斥鎖和自旋鎖」,有很多高級的鎖都是基于它們實現的,你可以認為它們是各種鎖的地基,所以我們必須清楚它倆之間的區別和應用。

加鎖的目的就是保證共享資源在任意時間里,只有一個線程訪問,這樣就可以避免多線程導致共享數據錯亂的問題。

當已經有一個線程加鎖后,其他線程加鎖則就會失敗,互斥鎖和自旋鎖對于加鎖失敗后的處理方式是不一樣的:

互斥鎖加鎖失敗后,線程會釋放 CPU,給其他線程;

自旋鎖加鎖失敗后,線程會忙等待,直到它拿到鎖;

互斥鎖是一種「獨占鎖」,比如當線程 A 加鎖成功后,此時互斥鎖已經被線程 A 獨占了,只要線程 A 沒有釋放手中的鎖,線程 B 加鎖就會失敗,于是就會釋放 CPU 讓給其他線程,既然線程 B 釋放掉了 CPU,自然線程 B 加鎖的代碼就會被阻塞。

對于互斥鎖加鎖失敗而阻塞的現象,是由操作系統內核實現的。當加鎖失敗時,內核會將線程置為「睡眠」狀態,等到鎖被釋放后,內核會在合適的時機喚醒線程,當這個線程成功獲取到鎖后,于是就可以繼續執行。如下圖:

所以,互斥鎖加鎖失敗時,會從用戶態陷入到內核態,讓內核幫我們切換線程,雖然簡化了使用鎖的難度,但是存在一定的性能開銷成本。

那這個開銷成本是什么呢?會有兩次線程上下文切換的成本:

當線程加鎖失敗時,內核會把線程的狀態從「運行」狀態設置為「睡眠」狀態,然后把 CPU 切換給其他線程運行;

接著,當鎖被釋放時,之前「睡眠」狀態的線程會變為「就緒」狀態,然后內核會在合適的時間,把 CPU 切換給該線程運行。

線程的上下文切換的是什么?當兩個線程是屬于同一個進程,因為虛擬內存是共享的,所以在切換時,虛擬內存這些資源就保持不動,只需要切換線程的私有數據、寄存器等不共享的數據。

上下切換的耗時有大佬統計過,大概在幾十納秒到幾微秒之間,如果你鎖住的代碼執行時間比較短,那可能上下文切換的時間都比你鎖住的代碼執行時間還要長。

所以,如果你能確定被鎖住的代碼執行時間很短,就不應該用互斥鎖,而應該選用自旋鎖,否則使用互斥鎖。

自旋鎖是通過 CPU 提供的CAS函數(Compare And Swap),在「用戶態」完成加鎖和解鎖操作,不會主動產生線程上下文切換,所以相比互斥鎖來說,會快一些,開銷也小一些。

一般加鎖的過程,包含兩個步驟:

第一步,查看鎖的狀態,如果鎖是空閑的,則執行第二步;

第二步,將鎖設置為當前線程持有;

CAS 函數就把這兩個步驟合并成一條硬件級指令,形成原子指令,這樣就保證了這兩個步驟是不可分割的,要么一次性執行完兩個步驟,要么兩個步驟都不執行。

使用自旋鎖的時候,當發生多線程競爭鎖的情況,加鎖失敗的線程會「忙等待」,直到它拿到鎖。這里的「忙等待」可以用while循環等待實現,不過最好是使用 CPU 提供的PAUSE指令來實現「忙等待」,因為可以減少循環等待時的耗電量。

自旋鎖是最比較簡單的一種鎖,一直自旋,利用 CPU 周期,直到鎖可用。需要注意,在單核 CPU 上,需要搶占式的調度器(即不斷通過時鐘中斷一個線程,運行其他線程)。否則,自旋鎖在單 CPU 上無法使用,因為一個自旋的線程永遠不會放棄 CPU。

自旋鎖開銷少,在多核系統下一般不會主動產生線程切換,適合異步、協程等在用戶態切換請求的編程方式,但如果被鎖住的代碼執行時間過長,自旋的線程會長時間占用 CPU 資源,所以自旋的時間和被鎖住的代碼執行的時間是成「正比」的關系,我們需要清楚的知道這一點。

自旋鎖與互斥鎖使用層面比較相似,但實現層面上完全不同:當加鎖失敗時,互斥鎖用「線程切換」來應對,自旋鎖則用「忙等待」來應對。

它倆是鎖的最基本處理方式,更高級的鎖都會選擇其中一個來實現,比如讀寫鎖既可以選擇互斥鎖實現,也可以基于自旋鎖實現。

讀寫鎖:讀和寫還有優先級區分?

讀寫鎖從字面意思我們也可以知道,它由「讀鎖」和「寫鎖」兩部分構成,如果只讀取共享資源用「讀鎖」加鎖,如果要修改共享資源則用「寫鎖」加鎖。

所以,讀寫鎖適用于能明確區分讀操作和寫操作的場景。

讀寫鎖的工作原理是:

當「寫鎖」沒有被線程持有時,多個線程能夠并發地持有讀鎖,這大大提高了共享資源的訪問效率,因為「讀鎖」是用于讀取共享資源的場景,所以多個線程同時持有讀鎖也不會破壞共享資源的數據。

但是,一旦「寫鎖」被線程持有后,讀線程的獲取讀鎖的操作會被阻塞,而且其他寫線程的獲取寫鎖的操作也會被阻塞。

所以說,寫鎖是獨占鎖,因為任何時刻只能有一個線程持有寫鎖,類似互斥鎖和自旋鎖,而讀鎖是共享鎖,因為讀鎖可以被多個線程同時持有。

知道了讀寫鎖的工作原理后,我們可以發現,讀寫鎖在讀多寫少的場景,能發揮出優勢。

另外,根據實現的不同,讀寫鎖可以分為「讀優先鎖」和「寫優先鎖」。

讀優先鎖期望的是,讀鎖能被更多的線程持有,以便提高讀線程的并發性,它的工作方式是:當讀線程 A 先持有了讀鎖,寫線程 B 在獲取寫鎖的時候,會被阻塞,并且在阻塞過程中,后續來的讀線程 C 仍然可以成功獲取讀鎖,最后直到讀線程 A 和 C 釋放讀鎖后,寫線程 B 才可以成功獲取讀鎖。如下圖:

而寫優先鎖是優先服務寫線程,其工作方式是:當讀線程 A 先持有了讀鎖,寫線程 B 在獲取寫鎖的時候,會被阻塞,并且在阻塞過程中,后續來的讀線程 C 獲取讀鎖時會失敗,于是讀線程 C 將被阻塞在獲取讀鎖的操作,這樣只要讀線程 A 釋放讀鎖后,寫線程 B 就可以成功獲取讀鎖。如下圖:

讀優先鎖對于讀線程并發性更好,但也不是沒有問題。我們試想一下,如果一直有讀線程獲取讀鎖,那么寫線程將永遠獲取不到寫鎖,這就造成了寫線程「饑餓」的現象。

寫優先鎖可以保證寫線程不會餓死,但是如果一直有寫線程獲取寫鎖,讀線程也會被「餓死」。

既然不管優先讀鎖還是寫鎖,對方可能會出現餓死問題,那么我們就不偏袒任何一方,搞個「公平讀寫鎖」。

公平讀寫鎖比較簡單的一種方式是:用隊列把獲取鎖的線程排隊,不管是寫線程還是讀線程都按照先進先出的原則加鎖即可,這樣讀線程仍然可以并發,也不會出現「饑餓」的現象。

互斥鎖和自旋鎖都是最基本的鎖,讀寫鎖可以根據場景來選擇這兩種鎖其中的一個進行實現。

樂觀鎖與悲觀鎖:做事的心態有何不同?

前面提到的互斥鎖、自旋鎖、讀寫鎖,都是屬于悲觀鎖。

悲觀鎖做事比較悲觀,它認為多線程同時修改共享資源的概率比較高,于是很容易出現沖突,所以訪問共享資源前,先要上鎖。

那相反的,如果多線程同時修改共享資源的概率比較低,就可以采用樂觀鎖。

樂觀鎖做事比較樂觀,它假定沖突的概率很低,它的工作方式是:先修改完共享資源,再驗證這段時間內有沒有發生沖突,如果沒有其他線程在修改資源,那么操作完成,如果發現有其他線程已經修改過這個資源,就放棄本次操作。

放棄后如何重試,這跟業務場景息息相關,雖然重試的成本很高,但是沖突的概率足夠低的話,還是可以接受的。

可見,樂觀鎖的心態是,不管三七二十一,先改了資源再說。另外,你會發現樂觀鎖全程并沒有加鎖,所以它也叫無鎖編程。

這里舉一個場景例子:在線文檔。

我們都知道在線文檔可以同時多人編輯的,如果使用了悲觀鎖,那么只要有一個用戶正在編輯文檔,此時其他用戶就無法打開相同的文檔了,這用戶體驗當然不好了。

那實現多人同時編輯,實際上是用了樂觀鎖,它允許多個用戶打開同一個文檔進行編輯,編輯完提交之后才驗證修改的內容是否有沖突。

怎么樣才算發生沖突?這里舉個例子,比如用戶 A 先在瀏覽器編輯文檔,之后用戶 B 在瀏覽器也打開了相同的文檔進行編輯,但是用戶 B 比用戶 A 提交改動,這一過程用戶 A 是不知道的,當 A 提交修改完的內容時,那么 A 和 B 之間并行修改的地方就會發生沖突。

服務端要怎么驗證是否沖突了呢?通常方案如下:

由于發生沖突的概率比較低,所以先讓用戶編輯文檔,但是瀏覽器在下載文檔時會記錄下服務端返回的文檔版本號;

當用戶提交修改時,發給服務端的請求會帶上原始文檔版本號,服務器收到后將它與當前版本號進行比較,如果版本號一致則修改成功,否則提交失敗。

實際上,我們常見的 SVN 和 Git 也是用了樂觀鎖的思想,先讓用戶編輯代碼,然后提交的時候,通過版本號來判斷是否產生了沖突,發生了沖突的地方,需要我們自己修改后,再重新提交。

樂觀鎖雖然去除了加鎖解鎖的操作,但是一旦發生沖突,重試的成本非常高,所以只有在沖突概率非常低,且加鎖成本非常高的場景時,才考慮使用樂觀鎖。

總結

開發過程中,最常見的就是互斥鎖的了,互斥鎖加鎖失敗時,會用「線程切換」來應對,當加鎖失敗的線程再次加鎖成功后的這一過程,會有兩次線程上下文切換的成本,性能損耗比較大。

如果我們明確知道被鎖住的代碼的執行時間很短,那我們應該選擇開銷比較小的自旋鎖,因為自旋鎖加鎖失敗時,并不會主動產生線程切換,而是一直忙等待,直到獲取到鎖,那么如果被鎖住的代碼執行時間很短,那這個忙等待的時間相對應也很短。

如果能區分讀操作和寫操作的場景,那讀寫鎖就更合適了,它允許多個讀線程可以同時持有讀鎖,提高了讀的并發性。根據偏袒讀方還是寫方,可以分為讀優先鎖和寫優先鎖,讀優先鎖并發性很強,但是寫線程會被餓死,而寫優先鎖會優先服務寫線程,讀線程也可能會被餓死,那為了避免饑餓的問題,于是就有了公平讀寫鎖,它是用隊列把請求鎖的線程排隊,并保證先入先出的原則來對線程加鎖,這樣便保證了某種線程不會被餓死,通用性也更好點。

互斥鎖和自旋鎖都是最基本的鎖,讀寫鎖可以根據場景來選擇這兩種鎖其中的一個進行實現。

另外,互斥鎖、自旋鎖、讀寫鎖都屬于悲觀鎖,悲觀鎖認為并發訪問共享資源時,沖突概率可能非常高,所以在訪問共享資源前,都需要先加鎖。

相反的,如果并發訪問共享資源時,沖突概率非常低的話,就可以使用樂觀鎖,它的工作方式是,在訪問共享資源時,不用先加鎖,修改完共享資源后,再驗證這段時間內有沒有發生沖突,如果沒有其他線程在修改資源,那么操作完成,如果發現有其他線程已經修改過這個資源,就放棄本次操作。

但是,一旦沖突概率上升,就不適合使用樂觀鎖了,因為它解決沖突的重試成本非常高。

不管使用的哪種鎖,我們的加鎖的代碼范圍應該盡可能的小,也就是加鎖的粒度要小,這樣執行速度會比較快。再來,使用上了合適的鎖,就會快上加快了。

責任編輯:xj

原文標題:面試官:你說說互斥鎖、自旋鎖、讀寫鎖、悲觀鎖、樂觀鎖的應用場景

文章出處:【微信公眾號:數據分析與開發】歡迎添加關注!文章轉載請注明出處。

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 編程
    +關注

    關注

    90

    文章

    3716

    瀏覽量

    97178
  • 程序員
    +關注

    關注

    4

    文章

    956

    瀏覽量

    30940
  • 線程
    +關注

    關注

    0

    文章

    509

    瀏覽量

    20825

原文標題:面試官:你說說互斥鎖、自旋鎖、讀寫鎖、悲觀鎖、樂觀鎖的應用場景

文章出處:【微信號:DBDevs,微信公眾號:數據分析與開發】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    面試必看:排隊自旋之MCS的實現原理與關鍵考點

    在并發編程面試中,“” 是繞不開的核心話題,而自旋作為輕量級的代表,其優化方案更是高頻考點。
    的頭像 發表于 02-09 16:51 ?746次閱讀
    面試必看:排隊自旋<b class='flag-5'>鎖</b>之MCS<b class='flag-5'>鎖</b>的實現原理與關鍵考點

    學習電子電路中常見的問題

    電子電路作為現代科技的基礎,其學習過程中常會遇到各種理論和實踐問題。無論是初學者還是有一定經驗的工程師,都可能面臨電路設計、元器件選型、信號處理等方面的困惑。本文將系統梳理電子電路學習中常見的典型
    的頭像 發表于 01-20 07:38 ?218次閱讀

    芯片編程器使用指南:如何避免芯片燒錄過程中的常見錯誤

    芯片燒錄失敗多源于細節疏漏,使用編程器需規避常見錯誤。首要確保芯片與編程器適配,核查封裝、電壓協議并驗證芯片 ID;重視環境與連接,做好靜電防護、保障電源穩定及觸點清潔;規范文件流程,嚴格版本核對
    的頭像 發表于 12-30 10:59 ?409次閱讀

    晶振使用中常見問題與解決方法

    一、頻偏造成的使用異常異常現象:色彩圖像不正常;音頻雜音,無數據傳輸,距離短,遙控無反應。常見處理:換一個就OK根本原因:晶振負載電容同電路不匹配。解決辦法:調整電路匹配電容大小,或換用不同負載電容
    的頭像 發表于 11-21 15:37 ?3997次閱讀
    晶振使用<b class='flag-5'>中常見</b>問題與解決方法

    正確的連接電機傳動軸的幾種常見連接方式及其優缺點

    在工業機械和自動化設備中,電機與傳動軸的連接方式直接影響設備的運行效率、穩定性和使用壽命。選擇合適的連接方式需要綜合考慮負載特性、安裝精度、維護便利性以及成本等因素。以下是幾種常見的電機傳動軸連接
    的頭像 發表于 09-10 07:41 ?3175次閱讀

    ?一文了智能門鎖常見幾種語音芯片方案

    智能是一種成熟且穩定的產品類型,它對語音芯片的要求,大致集中于以下幾點: 使用簡單且好打樣:由于智能大多依托方案公司開發,購買 pcba 回來組裝,產品類型豐富,語音需求多樣,頻繁打樣會影響開發
    的頭像 發表于 08-01 17:31 ?1139次閱讀

    什么是電磁控制板?24路控板的使用步驟概述

    電磁控制板是一種專用于控制電磁工作狀態的電子裝置,它可以接收指令并對電磁進行通斷電操作,從而實現對相關設備的鎖定和解鎖控制。電磁本身依靠電流產生磁場作用力來吸附或釋放
    的頭像 發表于 07-14 16:26 ?779次閱讀
    什么是電磁<b class='flag-5'>鎖</b>控制板?24路<b class='flag-5'>鎖</b>控板的使用步驟概述

    【HarmonyOS 5】鴻蒙中常見的標題欄布局方案

    【HarmonyOS 5】鴻蒙中常見的標題欄布局方案 ##鴻蒙開發能力 ##HarmonyOS SDK應用服務##鴻蒙金融類應用 (金融理財# 一、問題背景: 鴻蒙中常見的標題欄:矩形區域,左邊
    的頭像 發表于 07-11 18:30 ?868次閱讀
    【HarmonyOS 5】鴻蒙<b class='flag-5'>中常見</b>的標題欄布局方案

    一文讀懂Allegro先進磁性開關和存器

    開關或存器器件。文中詳細闡釋了區分 Allegro 開關與存器的關鍵參數,以助力設計師精準定位符合需求的器件。結論部分總結了選型流程要點,并梳理了 Allegro 開關和存器的常見
    的頭像 發表于 06-12 17:26 ?1899次閱讀
    一文讀懂Allegro先進磁性開關和<b class='flag-5'>鎖</b>存器

    PLC在使用過程中常見的技術故障分析及維護

    PLC(可編程邏輯控制器)在使用過程中常見的技術故障分析及維護措施如下: 一、常見技術故障分析 1. 外圍電路元器件故障 ● 故障描述:在PLC控制回路中,如果元器件損壞,PLC控制系統會立即自動
    的頭像 發表于 04-23 17:06 ?1642次閱讀
    PLC在使用過程<b class='flag-5'>中常見</b>的技術故障分析及維護

    電子電路設計中常用的接地方式詳解

    在電子電路設計中,接地方式的選擇至關重要,它直接影響到電路的穩定性、抗干擾能力和安全性。以下是電子電路設計中常用的幾種接地方式的詳解: 一、浮地 1. 定義:浮地是指電路或設備與公共地線可能引起環流
    的頭像 發表于 04-17 16:24 ?2063次閱讀
    電子電路設計<b class='flag-5'>中常</b>用的接地方式詳解

    工程繪圖CAD使用中常見的問題

    CAD(計算機輔助設計)在使用過程中,用戶可能會遇到多種常見問題。以下是一些常見的問題及其解決辦法: 一、軟件操作問題 1. Ctrl+N無效 ? ?● 問題描述:Ctrl+N通常是新建命令,但有
    的頭像 發表于 04-14 15:02 ?1134次閱讀

    STM32的幾種開發方式,你都知道嗎?(可下載)

    經常有人會問,你們 STM32 編程是用庫函數 還是用寄存器的...會說庫函數方便,容 易,都用庫函數...等等這樣的問題,今天我們就來 dis 一下這幾種編程方式,STM32 編程
    發表于 04-03 11:24 ?0次下載

    PCBA加工返修全攻略:常見問題一網打盡

    一站式PCBA智造廠家今天為大家講講PCBA加工返修中常見的問題有哪些?PCBA返修常見問題及解決方案。在PCBA加工過程中,產品的質量直接影響著客戶的滿意度,而返修問題是客戶尤為關注的焦點。盡管
    的頭像 發表于 04-02 18:00 ?976次閱讀

    常見網絡負載均衡的幾種方式

    常見網絡負載均衡的幾種方式包括:DNS負載均衡、反向代理負載均衡、IP負載均衡、應用層負載均衡、鏈路層負載均衡。以下是小編對幾種常見的網絡負載均衡方式及其詳細展開介紹。
    的頭像 發表于 03-06 11:14 ?1453次閱讀