java - 讀寫鎖中 寫鎖的降級(jí)問(wèn)題
問(wèn)題描述
寫鎖降級(jí)的時(shí)候 要求當(dāng)前已經(jīng)獲取到寫鎖的線程把持住寫鎖,然后獲取讀鎖,隨后釋放寫鎖。
那么為什么要求把持住寫鎖呢??
上圖中提到 “ 那么當(dāng)前線程無(wú)法感知線程T的數(shù)據(jù)更新 ” 這個(gè)如何理解。如果線程T獲取到了寫鎖,然后對(duì)數(shù)據(jù)進(jìn)行了更新,在沒(méi)有釋放寫鎖之前,當(dāng)前線程無(wú)法感知到數(shù)據(jù)的更新,這個(gè)是沒(méi)問(wèn)題的吧,但是一旦線程T釋放 寫鎖,那么當(dāng)前線程就可以感知到數(shù)據(jù)的更新了,這里理解應(yīng)該沒(méi)問(wèn)題吧?
help me to explain,thx
問(wèn)題解答
回答1:這是在解釋鎖降級(jí)吧,首先寫鎖是獨(dú)占的,讀鎖是共享的,然后讀寫鎖是線程間互斥的,鎖降級(jí)的前提是所有線程都希望對(duì)數(shù)據(jù)變化敏感,但是因?yàn)閷戞i只有一個(gè),所以會(huì)發(fā)生降級(jí)。如果先釋放寫鎖,再獲取讀鎖,可能在獲取之前,會(huì)有其他線程獲取到寫鎖,阻塞讀鎖的獲取,就無(wú)法感知數(shù)據(jù)變化了。所以需要先hold住寫鎖,保證數(shù)據(jù)無(wú)變化,獲取讀鎖,然后再釋放寫鎖。
回答2:寫鎖不允許其他進(jìn)程的讀寫操作,讀鎖的允許讀操作。就按你抓圖的例子來(lái)看,當(dāng)你釋放寫鎖后,T進(jìn)程獲取了寫鎖,這時(shí)你就無(wú)法獲取讀鎖了,所以要先獲取讀鎖后,再釋放寫鎖。
首先你沒(méi)理解讀寫鎖的意義,讀鎖的存在意味著不允許其他寫操作的存在。按照你提供的例子,可能存在一個(gè)事務(wù)線程不希望自己的操作被別的線程中斷,而這個(gè)事務(wù)操作可能分成多部分操作更新不同的數(shù)據(jù)(或表)甚至非常耗時(shí)。如果長(zhǎng)時(shí)間用寫鎖獨(dú)占,顯然對(duì)于某些高響應(yīng)的應(yīng)用是不允許的,所以在完成部分寫操作后,退而使用讀鎖降級(jí),來(lái)允許響應(yīng)其他進(jìn)程的讀操作。只有當(dāng)全部事務(wù)完成后才真正釋放鎖。按你的理解如果當(dāng)中寫鎖被其他線程占用,那么這個(gè)事務(wù)線程將不得不中斷等待別的寫鎖釋放。
回答3:“ 那么當(dāng)前線程無(wú)法感知線程T的數(shù)據(jù)更新 ”
T1 讀取 數(shù)據(jù) Data時(shí),將數(shù)據(jù)值復(fù)制到當(dāng)前線程的上下文,因此其他線程沒(méi)有辦法感知數(shù)據(jù)是否更新。只要當(dāng)數(shù)據(jù)提交內(nèi)存中,即Heap中,其他線程才可以得到最新值。
相關(guān)文章:
1. javascript - node.js promise沒(méi)用2. android 如何實(shí)現(xiàn)如圖中的鍵盤上的公式及edittext的內(nèi)容展示呢3. c++ - 如何正確的使用QWebEngineView?4. golang - 用IDE看docker源碼時(shí)的小問(wèn)題5. javascript - js 寫一個(gè)正則 提取文本中的數(shù)據(jù)6. 算法 - python 給定一個(gè)正整數(shù)a和一個(gè)包含任意個(gè)正整數(shù)的 列表 b,求所有<=a 的加法組合7. yii2中restful配置好后在nginx下報(bào)404錯(cuò)誤8. java - 我在用Struts2上傳文件時(shí),報(bào)以下錯(cuò)誤怎么回事?9. PHP注冊(cè)功能10. php - 注冊(cè)驗(yàn)證郵箱失效后操作問(wèn)題
