淺談一下Java中的ReentrantLock
一、ReentrantLock是用來做什么的?
這個類是JUC工具包中對線程安全問題提供的一種解決方案,它主要是用來給對象上鎖,保證同一時間這能有一個線程在訪問當前對象。這樣處理是為了防止如果一個線程對某個公共變量進行了改變,而其它線程讀取時讀出來的是原有數(shù)據(jù)導致臟讀的問題。
二、ReentrantLock的實現(xiàn)原理是什么?
ReentrantLock主要是通過同步隊列和CAS機制來實現(xiàn)的,它實現(xiàn)的過程中主要包含下面幾個屬性:
- status:鎖狀態(tài),0表示沒有線程獲取鎖,1表示已有線程獲取鎖
- exclusiveOwnerThread:當前持有鎖的線程
- Node:節(jié)點,是ReentrantLock內(nèi)部維持的一個雙向鏈表(同步阻塞隊列)的基本構成
具體流程
1、上鎖更新鎖狀態(tài):當A線程持有鎖時,會通過CAS將status狀態(tài)置為1,并將A線程自身存入exclusiveOwnerThread屬性當中;
2、線程入隊:而后線程B通過CAS獲取鎖時發(fā)現(xiàn)無法獲取鎖,此時就會獲取node信息,但是由于node雙向鏈表是null所以會通過CAS來創(chuàng)建一個雙向鏈表的head對象,之后再把線程B封裝成的Node節(jié)點通過尾插法接入雙向鏈表的尾部;
3、線程阻塞:入隊完成后再調(diào)用park方法進行阻塞;
4、釋放鎖并更新鎖狀態(tài):當A線程釋放鎖時會將status屬性重置為0,且把exclusiveOwnerThread置為null;
5、喚醒線程:A線程釋放鎖完畢后會調(diào)用unpark方法來喚醒雙向鏈表中下一節(jié)點的線程B;
三、ReentrantLock對比于Synchronized有哪些優(yōu)缺點?
1、ReentrantLock的鎖狀態(tài)是可見的,而Synchronized的鎖狀態(tài)是不可見的;
2、ReentrantLock是JDK層面的實現(xiàn),而Synchronized是JVM層面的實現(xiàn);
3、ReentrantLock需要手動釋放鎖,而Synchronized不需要手動釋放鎖;
4、ReentrantLock可以是非公平鎖也可以是公平鎖,而Synchronized只能是非公平鎖;
5、ReentrantLock是可被中斷的,而Synchronized是不可悲中斷的;
6、Synchronized在特定條件下是后來的線程先獲取鎖,而ReentrantLock是先來的線程先獲取鎖;
四、ReentrantLock的簡單使用
static ReentrantLock lock = new ReentrantLock(); for (int i = 0; i < 10; i++) { Thread thread = new Thread(() -> { lock.lock(); try { for (int j = 0; j < 10000; j++) { sum++; } } finally { lock.unlock(); latch.countDown(); } }); thread.start(); }
到此這篇關于淺談一下Java中的ReentrantLock的文章就介紹到這了,更多相關淺談ReentrantLock內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
SpringBoot集成WebSocket長連接實際應用詳解
這篇文章主要介紹了SpringBoot集成WebSocket長連接實際應用詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-06-06Java使用CompletableFuture進行非阻塞IO詳解
這篇文章主要介紹了Java使用CompletableFuture進行非阻塞IO詳解,CompletableFuture是Java中的一個類,用于支持異步編程和處理異步任務的結(jié)果,它提供了一種方便的方式來處理異步操作,并允許我們以非阻塞的方式執(zhí)行任務,需要的朋友可以參考下2023-09-09JavaWeb利用struts實現(xiàn)文件下載時改變文件名稱
這篇文章主要為大家詳細介紹了JavaWeb利用struts實現(xiàn)文件下載時改變文件名稱的相關資料,需要的朋友可以參考下2016-06-06