0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Linux讀寫(xiě)鎖邏輯解析—嘗試獲取寫(xiě)鎖

冬至子 ? 來(lái)源:內(nèi)核工匠 ? 作者:郭健Cojack ? 2023-12-04 11:12 ? 次閱讀

Rwsem的count成員還有一些bit用來(lái)標(biāo)記當(dāng)前讀寫(xiě)鎖狀態(tài)(waiter bit和handoff bit),也需要根據(jù)情況進(jìn)行調(diào)整:

1.jpg

A、如果等待隊(duì)列為空了,肯定是要清除waiter flag,同時(shí)要清除handoff flag,畢竟沒(méi)有什么等待任務(wù)可以遞交鎖了。

B、雖然隊(duì)列非空,但已經(jīng)喚醒了reader,那么需要清除handoff標(biāo)記,畢竟top waiter已經(jīng)被喚醒去持鎖了,完成了鎖的遞交。

C、完成sem->count的調(diào)整

第二輪將喚醒的reader加入喚醒隊(duì)列,具體的邏輯如下:

1.jpg

主要是把等待任務(wù)對(duì)象的task成員設(shè)置為NULL,喚醒之后根據(jù)這個(gè)成員來(lái)判斷是正常喚醒還是異常喚醒路徑。

這里對(duì)喚醒等待隊(duì)列上的reader和writer處理是不一樣的。對(duì)于writer,喚醒之然后被調(diào)度到之后再去試圖持鎖。對(duì)于reader,在喚醒路徑上就已經(jīng)持鎖(增加rwsem的reader count,并且修改了相關(guān)的狀態(tài)標(biāo)記)。之所以這么做主要是降低調(diào)度的開(kāi)銷,畢竟若干個(gè)reader線程被喚醒之后,獲得CPU資源再去持鎖,持鎖失敗然后繼續(xù)阻塞,這些都會(huì)增加調(diào)度的負(fù)載。

七、嘗試獲取寫(xiě)鎖

和down_write不一樣,down_write_trylock只是嘗試獲取寫(xiě)鎖,如果成功,那么自然是好的,直接返回1,如果失敗,也不會(huì)阻塞,只是返回0就可以了。代碼主邏輯在rwsem_write_trylock函數(shù)中,如下:

1.jpg

tmp的初始值設(shè)定為RWSEM_UNLOCKED_VALUE(0值),對(duì)于writer而言,只有rwsem是空鎖的時(shí)候才能進(jìn)入臨界區(qū)。如果當(dāng)前的sem->count等于0,那么給sem->count賦值RWSEM_WRITER_LOCKED,標(biāo)記持鎖成功,并且把owner設(shè)定為當(dāng)前task。

atomic_long_try_cmpxchg_acquire函數(shù)有三個(gè)參數(shù),從左到右分別是value,old和new。該函數(shù)會(huì)對(duì)比value和old,如果相等那么執(zhí)行賦值value=new同時(shí)返回true。如果不相等,不執(zhí)行賦值操作,直接返回false。

八、獲取寫(xiě)鎖

Writer獲取寫(xiě)鎖的代碼主要在__down_write_common函數(shù)中,如下:

1.jpg

rwsem_write_trylock(快速路徑)上一節(jié)已經(jīng)描述,我們主要看慢速路徑的邏輯(樂(lè)觀自旋我們下面會(huì)講,這里暫且略過(guò)):

1.jpg

首先準(zhǔn)備好一個(gè)等待任務(wù)對(duì)象(棧上)并初始化,將其掛入等待隊(duì)列。在真正睡眠之前,我們需要做一些喚醒動(dòng)作(和reader持鎖過(guò)程類似,有可能在掛入等待隊(duì)列的時(shí)候,臨界區(qū)線程恰好離開(kāi),變成空鎖),具體邏輯如下:

1.jpg

A、如果我們是等待隊(duì)列的top waiter(等待隊(duì)列從空變?yōu)榉强眨?,那么需要設(shè)定RWSEM_FLAG_WAITERS標(biāo)記,直接進(jìn)入后續(xù)阻塞邏輯。如果不是,那么邏輯要復(fù)雜點(diǎn),需要掃描一下之前掛入隊(duì)列的任務(wù),看看是否需要喚醒。

B、如果是writer持鎖,那么不需要任何喚醒動(dòng)作,畢竟writer是排他的

C、如果是空鎖狀態(tài),我們需要喚醒top waiter(RWSEM_WAKE_ANY,top writer或者reader們)。你可能會(huì)疑問(wèn):為何空鎖還要喚醒等待隊(duì)列的線程?當(dāng)前線程快馬加鞭去持鎖不就OK了嗎?這主要是和handoff邏輯相關(guān),這時(shí)候更應(yīng)該持鎖的是等待隊(duì)列中設(shè)置了handoff的那個(gè)waiter,而不是當(dāng)前writer。如果是reader在臨界區(qū)內(nèi),那么,我們將喚醒本等待隊(duì)列頭部的所有reader(RWSEM_WAKE_READERS)。

D、上面僅僅是標(biāo)記喚醒者,這里的代碼段完成具體的喚醒動(dòng)作

下面進(jìn)入具體writer的阻塞過(guò)程:

1.jpg

A、調(diào)用rwsem_try_write_lock試圖持鎖,如果成功持鎖則退出循環(huán),不再阻塞。有兩個(gè)邏輯路徑會(huì)路過(guò)這里。一個(gè)是線程持鎖失敗進(jìn)入這里,另外一個(gè)是阻塞后被喚醒試圖持鎖。

B、有pending的信號(hào),異常路徑退出

C、持鎖失敗但是設(shè)置了handoff,那么該線程對(duì)owner進(jìn)行自旋等待,以便加快鎖的傳遞。

D、進(jìn)入阻塞狀態(tài)

E、喚醒之后,重新試圖持鎖。Writer和reader不一樣,writer是喚醒之后自己再通過(guò)rwsem_try_write_lock試圖持鎖,而reader是在喚醒路徑上持鎖。

rwsem_try_write_lock代碼如下:

1.jpg

A、如果已經(jīng)設(shè)置了handoff,并且自己不是top waiter(top waiter才是鎖要遞交的對(duì)象),返回false,持鎖失敗。如果是top waiter,那么就設(shè)置handoff_set,標(biāo)記自己就是鎖遞交的目標(biāo)任務(wù)。

B、如果當(dāng)前rwsem已經(jīng)有了owner,那么說(shuō)明該鎖被偷走了。在適當(dāng)?shù)臈l件下(等待超時(shí))設(shè)置handoff標(biāo)記,防止后續(xù)繼續(xù)被搶。如果已經(jīng)設(shè)置了handoff就不必重復(fù)設(shè)置了。

C、如果當(dāng)前rwsem沒(méi)有owner,則持鎖成功,清除handoff標(biāo)記并根據(jù)情況設(shè)置waiter標(biāo)記。

D、通過(guò)原子操作來(lái)持鎖,成功操作后退出循環(huán),否則是有其他線程插入,需要重復(fù)上面的邏輯。

1.jpg

至此我們要不獲取了鎖并清除了handoff bit(B邏輯塊),或者沒(méi)有獲取鎖,僅僅是設(shè)置了handoff bit(A邏輯塊)。

九、釋放寫(xiě)鎖

除了清除了owner task成員,其他邏輯和釋放讀鎖類似,不再贅述。

十、樂(lè)觀自旋的條件

只有writer在進(jìn)入慢速路徑的時(shí)候才會(huì)進(jìn)行樂(lè)觀自旋,而rwsem_can_spin_on_owner函數(shù)用來(lái)判斷writer是否可以樂(lè)觀自旋:

1.jpg

A、本cpu上需要reschedule,還自旋個(gè)毛線,趕緊去睡眠也順便觸發(fā)一次調(diào)度

B、讀取sem->owner,標(biāo)記部分保存在flags臨時(shí)變量中,任務(wù)指針保存在owner中

C、如果該rwsem已經(jīng)禁止了對(duì)應(yīng)的nonspinnable標(biāo)志,那么肯定是不能樂(lè)觀自旋了。如果當(dāng)前rwsem沒(méi)有禁止,那么需要看看owner的狀態(tài)。這里需要特別說(shuō)明的是:為了方便debug,我們?cè)卺尫抛x鎖的時(shí)候并不會(huì)清除owner task。也就是說(shuō),對(duì)于reader而言,owner中的task信息是最后進(jìn)入臨界區(qū)的那個(gè)reader,僅此而已,實(shí)際這個(gè)task可能已經(jīng)離開(kāi)臨界區(qū),甚至已經(jīng)銷毀都有可能。所以,如果rwsem是reader擁有,那么其實(shí)判斷owner是否在cpu上運(yùn)行是沒(méi)有意義的,因此owner是reader的話是允許進(jìn)行樂(lè)觀自旋的(ret的缺省值是true),通過(guò)超時(shí)來(lái)控制自旋的退出。如果rwsem是writer擁有,那么owner的的確確是正在持鎖的線程,如果該線程沒(méi)有在CPU上運(yùn)行(不能很快離開(kāi)臨界區(qū)),那么也不能樂(lè)觀自旋。

十一、rwsem_spin_on_owner

函數(shù)rwsem_spin_on_owner的功能是對(duì)rwsem的owner task進(jìn)行樂(lè)觀自旋(即不斷輪詢其狀態(tài),僅writer有效),詳細(xì)的代碼邏輯如下:

1.jpg

A、在自旋之前,首先要獲得初始的狀態(tài)(owner task指針以及2-bit LSB flag),當(dāng)這些狀態(tài)發(fā)生變化才好退出自旋。

B、rwsem_owner_state函數(shù)會(huì)根據(jù)當(dāng)前的owner task和flag判斷當(dāng)前的owner state。owner state的狀態(tài)總結(jié)如下:

1.jpg

只有明確的知道當(dāng)前rwsem的owner是某個(gè)writer線程且沒(méi)有禁止自旋的時(shí)候才開(kāi)啟下面的自旋過(guò)程。對(duì)于其他情況,例如reader owned的場(chǎng)景,我們不需要spin on owner,直接返回。

C、只要owner task或者flag其一發(fā)生變化,這里就會(huì)停止輪詢,同時(shí)也會(huì)返回當(dāng)前的狀態(tài),說(shuō)明停止自旋的原因。例如當(dāng)owner task(一定是writer)離開(kāi)臨界區(qū)的時(shí)候會(huì)清空rwsem的owner域(owner task和flag會(huì)清零),這時(shí)候自旋的writer會(huì)停止自旋,到外層函數(shù)會(huì)去試圖持鎖。當(dāng)然也有可能是其他自旋writer搶到了鎖,owner task從A切到B。無(wú)論那種情況,統(tǒng)一終止對(duì)owner的自旋。

D、如果當(dāng)前cpu需要reschedule或者owner task沒(méi)有正在運(yùn)行,那么也需要停止自旋

十二、Writer的樂(lè)觀自旋

和mutex的樂(lè)觀自旋的概念是類似的,想要進(jìn)行rwsem的樂(lè)觀自旋,首先要獲取osq鎖,只有獲得了osq lock才能進(jìn)入rwsem的樂(lè)觀自旋,否則自旋在per cpu的mcs lock上。Writer通過(guò)rwsem_optimistic_spin完成整個(gè)樂(lè)觀自旋的過(guò)程。對(duì)于writer owned場(chǎng)景,自旋發(fā)生在rwsem_spin_on_owner中,上一節(jié)已經(jīng)描述了,這里我們主要看reader owned的情況,這時(shí)候通過(guò)for loop不斷自旋去持鎖:

1.jpg

2.jpg

A、對(duì)于rwsem,只有writer-owned場(chǎng)景能清楚的知道owner task是哪一個(gè)。因此,如果是writer-owned場(chǎng)景,會(huì)在rwsem_spin_on_owner函數(shù)進(jìn)行自旋。對(duì)于非writer-owned場(chǎng)景(reader-owned場(chǎng)景或者禁止了樂(lè)觀自旋),在rwsem_spin_on_owner函數(shù)中會(huì)直接返回。從rwsem_spin_on_owner函數(shù)返回會(huì)給出owner state,如果需要退出樂(lè)觀自旋,那么這里break掉,自旋失敗,下面就準(zhǔn)備掛入等待隊(duì)列了。

B、每次退出rwsem_spin_on_owner并且沒(méi)有要退出自旋的時(shí)候,都試著去獲取rwsem,如果持鎖成功那么退出樂(lè)觀自旋。

C、C和D是對(duì)reader-owned場(chǎng)景的處理。每次rwsem的owner state發(fā)生變化(從non-reader變成reader-owned狀態(tài))時(shí)都會(huì)重新初始化 rspin_threshold。

D、Owner state沒(méi)有發(fā)生變化,那么當(dāng)前試圖持鎖的writer可以進(jìn)行樂(lè)觀自旋,但是需要有一個(gè)度,畢竟rwsem的臨界區(qū)內(nèi)可能有多個(gè)reader線程,這有可能使得writer樂(lè)觀自旋很長(zhǎng)時(shí)間。設(shè)置自旋門限閾值的公式是Spinning threshold = (10 + nr_readers/2)us,最大25us(30 reader)。一旦自旋超期,那么將調(diào)用rwsem_set_nonspinnable禁止樂(lè)觀自旋。

E、對(duì)于writer-owned場(chǎng)景,need_resched在函數(shù)rwsem_spin_on_owner中完成,對(duì)于reader-owned場(chǎng)景,也是需要檢查owner task所在cpu的resched情況。畢竟當(dāng)前任務(wù)如果有調(diào)度需求,無(wú)論reader持鎖還是writer持鎖場(chǎng)景都要停止自旋。

F、在reader-owned場(chǎng)景中,由于無(wú)法判定臨界區(qū)reader們的執(zhí)行狀態(tài),因此rt線程的樂(lè)觀自旋需要更加的謹(jǐn)慎,畢竟有可能自旋的rt線程和臨界區(qū)的reader在一個(gè)CPU上從而導(dǎo)致活鎖現(xiàn)象。當(dāng)然也不能禁止rt線程的自旋,畢竟在臨界區(qū)為空的情況下,rt自旋會(huì)有一定的收益的。允許rt線程自旋的場(chǎng)景有兩個(gè):

a) lock owner正在釋放鎖,sem->owner被清除但是鎖還沒(méi)有釋放。

b) 鎖是空閑的并且sem->owner已清除,但是在我們嘗試獲取鎖之前另一個(gè)任務(wù)剛剛進(jìn)入并獲取了鎖(例如一個(gè)自旋的writer先于我們進(jìn)入臨界區(qū))。

十三、關(guān)于handoff

1、設(shè)置handoff標(biāo)記

設(shè)置handoff往往是發(fā)生在喚醒持鎖階段。對(duì)于等待隊(duì)列的writer,喚醒之后要調(diào)度執(zhí)行后才去持鎖,這是一個(gè)長(zhǎng)路徑,很可能被其他的write或者reader把鎖搶走。喚醒等待隊(duì)列中的reader們有點(diǎn)不一樣,在喚醒路徑上就會(huì)從這一組待喚醒的reader們選出一個(gè)代表(一般是top waiter)去持鎖,然后再一個(gè)個(gè)的喚醒。在這個(gè)reader代表線程持鎖的時(shí)候也有可能由于writer偷鎖而失?。╮eader雖然也會(huì)偷鎖,但是偷鎖的reader也會(huì)喚醒等待隊(duì)列的reader們,完成top waiter未完成的工作)。

無(wú)論是reader還是writer,如果喚醒后持鎖失敗,并且等待時(shí)間已經(jīng)超過(guò)了RWSEM_WAIT_TIMEOUT,這時(shí)候就會(huì)設(shè)置handoff bit,防止等待隊(duì)列的waiter餓死。具體設(shè)置handoff bit的場(chǎng)景如下:

1.jpg

2、清除handoff標(biāo)記

標(biāo)記了hand off之后,快速路徑、樂(lè)觀偷鎖(reader)、樂(lè)觀自旋(writer)都無(wú)法完成持鎖,鎖最終會(huì)遞交給top waiter的線程,完成持鎖。一旦完成持鎖,handoff標(biāo)記就會(huì)被清除。具體清除handoff bit的場(chǎng)景包括:

1.jpg

3、確保鎖的所有權(quán)遞交給top waiter

1.jpg

十四、結(jié)論

標(biāo)準(zhǔn)linux內(nèi)核的讀寫(xiě)鎖是在公平性、吞吐量和延遲選擇了比較均衡的策略,這樣的策略在手機(jī)平臺(tái)上(特別是重載場(chǎng)景下)不能算是“優(yōu)秀”,只能是合格吧。實(shí)際上,在手機(jī)用戶交互場(chǎng)景中,我們更期望是確保用戶體驗(yàn)相關(guān)線程的持鎖時(shí)延,同時(shí)兼顧吞吐量。在這樣的背景下,OPPO內(nèi)核團(tuán)隊(duì)對(duì)linux中的讀寫(xiě)鎖進(jìn)行了優(yōu)化,下一次有機(jī)會(huì)可以分享我們?cè)谧x寫(xiě)鎖的持鎖時(shí)延方面做的改進(jìn)。

聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • cpu
    cpu
    +關(guān)注

    關(guān)注

    68

    文章

    10896

    瀏覽量

    212512
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11332

    瀏覽量

    210023
  • 狀態(tài)機(jī)
    +關(guān)注

    關(guān)注

    2

    文章

    492

    瀏覽量

    27607
  • Spin
    +關(guān)注

    關(guān)注

    0

    文章

    4

    瀏覽量

    8034
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    Linux下線程間通訊---讀寫(xiě)和條件變量

    讀寫(xiě),它把對(duì)共享資源的訪問(wèn)者劃分成讀者和寫(xiě)者,讀者只對(duì)共享資源進(jìn)行讀訪問(wèn),寫(xiě)者則需要對(duì)共享資源進(jìn)行寫(xiě)操作。件變量是線程可用的一種同步機(jī)制,
    的頭像 發(fā)表于 08-26 20:44 ?1528次閱讀
    <b class='flag-5'>Linux</b>下線程間通訊---<b class='flag-5'>讀寫(xiě)</b><b class='flag-5'>鎖</b>和條件變量

    Linux讀寫(xiě)邏輯解析Linux為何會(huì)引入讀寫(xiě)?

    除了mutex,在linux內(nèi)核中,還有一個(gè)經(jīng)常用到的睡眠就是rw semaphore(后文簡(jiǎn)稱為rwsem),它到底和mutex有什么不同呢?
    的頭像 發(fā)表于 12-04 11:04 ?988次閱讀
    <b class='flag-5'>Linux</b><b class='flag-5'>讀寫(xiě)</b><b class='flag-5'>鎖</b><b class='flag-5'>邏輯</b><b class='flag-5'>解析</b>—<b class='flag-5'>Linux</b>為何會(huì)引入<b class='flag-5'>讀寫(xiě)</b><b class='flag-5'>鎖</b>?

    寫(xiě)FPGA代碼時(shí),產(chǎn)生了存器有什么影響嗎

    經(jīng)常看到各種寫(xiě)HDL代碼時(shí)說(shuō)要避免生成存器,但是在某些情況,我不關(guān)心那種情況即使它生成了存器,對(duì)我的工程實(shí)現(xiàn)也沒(méi)有什么影響啊,想請(qǐng)教下各位大神,既然這樣,為什么還要避免生成存器(
    發(fā)表于 01-08 23:54

    Lock體系結(jié)構(gòu)和讀寫(xiě)機(jī)制解析

    問(wèn)題,JDK中還有另一套讀寫(xiě)機(jī)制。讀寫(xiě)中維護(hù)一個(gè)共享讀和一個(gè)排它寫(xiě)
    發(fā)表于 01-05 17:53

    《有》/《無(wú)》/《簽約》/《解鎖》/《越獄》/《激活》專

    《有》/《無(wú)》/《簽約》/《解鎖》/《越獄》/《激活》專業(yè)技術(shù)詞解析 在討論區(qū)里,大家看到:《有版》,《無(wú)版》,《解
    發(fā)表于 02-03 11:05 ?960次閱讀

    Linux 自旋spinlock

    背景 由于在多處理器環(huán)境中某些資源的有限性,有時(shí)需要互斥訪問(wèn)(mutual exclusion),這時(shí)候就需要引入的概念,只有獲取的任務(wù)才能夠?qū)Y源進(jìn)行訪問(wèn),由于多線程的核心是CPU的時(shí)間分片
    的頭像 發(fā)表于 09-11 14:36 ?2107次閱讀

    詳談Linux操作系統(tǒng)的三種狀態(tài)的讀寫(xiě)

    讀寫(xiě)是另一種實(shí)現(xiàn)線程間同步的方式。與互斥量類似,但讀寫(xiě)將操作分為讀、寫(xiě)兩種方式,可以多個(gè)線程同時(shí)占用讀模式的
    的頭像 發(fā)表于 09-27 14:57 ?3142次閱讀

    Linux中的傷害/等待互斥介紹

    序言:近期讀Linux 5.15的發(fā)布說(shuō)明,該版本合并了實(shí)時(shí)機(jī)制,當(dāng)開(kāi)啟配置宏CONFIG_PREEMPT_RT的時(shí)候,這些被基于實(shí)時(shí)互斥的變體替代:mutex、ww_mutex
    的頭像 發(fā)表于 11-06 17:27 ?2699次閱讀

    使用Linux自旋實(shí)現(xiàn)互斥點(diǎn)燈

    自旋最多只能被一個(gè)可執(zhí)行線程持有。如果一個(gè)線程試圖獲得一個(gè)已經(jīng)被持有的自旋,那么該線程將循環(huán)等待,然后不斷的判斷是否能夠被成功獲取,直到獲取
    的頭像 發(fā)表于 04-13 15:09 ?804次閱讀
    使用<b class='flag-5'>Linux</b>自旋<b class='flag-5'>鎖</b>實(shí)現(xiàn)互斥點(diǎn)燈

    Linux實(shí)例:多線程和互斥到底該如何使用

    最近在寫(xiě)多進(jìn)程和Linux中的各種的文章,總覺(jué)得只有文字講解雖然能夠知道多進(jìn)程和互斥是什么,但是還是不知道到底該怎么用。
    發(fā)表于 05-18 14:16 ?432次閱讀
    <b class='flag-5'>Linux</b>實(shí)例:多線程和互斥<b class='flag-5'>鎖</b>到底該如何使用

    Linux互斥的作用 互斥是什么

    。如果釋放互斥時(shí)有一個(gè)以上的線程阻塞,那么這些阻塞的線程會(huì)被喚醒,它們都會(huì)嘗試對(duì)互斥進(jìn)行加鎖,當(dāng)有一個(gè)線程成功對(duì)互斥鎖上鎖之后,其它線程就不能再次上鎖了,只能再次陷入阻塞,等待下一次解鎖。 初始化互斥
    的頭像 發(fā)表于 07-21 11:13 ?978次閱讀

    自旋和互斥的區(qū)別有哪些

    之間的區(qū)別: 實(shí)現(xiàn)方式上的區(qū)別:互斥是基于自旋而實(shí)現(xiàn)的,所以自旋鎖相較于互斥更加底層; 開(kāi)銷上的區(qū)別:獲取不到互斥
    的頭像 發(fā)表于 07-21 11:19 ?9533次閱讀

    讀寫(xiě)的實(shí)現(xiàn)原理規(guī)則

    讀寫(xiě) 互斥或自旋要么是加鎖狀態(tài)、要么是不加鎖狀態(tài),而且一次只有一個(gè)線程可以對(duì)其加鎖。 讀寫(xiě)
    的頭像 發(fā)表于 07-21 11:21 ?940次閱讀
    <b class='flag-5'>讀寫(xiě)</b><b class='flag-5'>鎖</b>的實(shí)現(xiàn)原理規(guī)則

    AQS獨(dú)占獲取

    AQS提供了兩種,獨(dú)占和共享。獨(dú)占只有一把,同一時(shí)間只允許一個(gè)線程獲得;而共享
    的頭像 發(fā)表于 10-13 14:51 ?490次閱讀
    AQS獨(dú)占<b class='flag-5'>鎖</b>的<b class='flag-5'>獲取</b>

    互斥和自旋的區(qū)別 自旋臨界區(qū)可以被中斷嗎?

    獲得了互斥時(shí),其他線程如果要獲取,則必須等待直到該線程釋放。互斥的實(shí)現(xiàn)通常會(huì)利用操作系統(tǒng)提供的原子操作和線程調(diào)度機(jī)制。當(dāng)某個(gè)線程
    的頭像 發(fā)表于 11-22 17:41 ?872次閱讀