ReadWriteLock介面 讀寫鎖維護一對關聯鎖,一個只用於讀操作,一個只用於寫操作。讀鎖可以由多個線程同時持有,又稱共用鎖。寫鎖同一時間只能由一個線程持有,又稱互斥鎖。同一時間,兩把鎖不能被不同線程持有。讀寫鎖適合讀取操作多於寫入操作的場景,改進互斥鎖的性能,比如集合的併發安全性改造,緩存組件 ...
ReadWriteLock介面
讀寫鎖維護一對關聯鎖,一個只用於讀操作,一個只用於寫操作。讀鎖可以由多個線程同時持有,又稱共用鎖。寫鎖同一時間只能由一個線程持有,又稱互斥鎖。同一時間,兩把鎖不能被不同線程持有。讀寫鎖適合讀取操作多於寫入操作的場景,改進互斥鎖的性能,比如集合的併發安全性改造,緩存組件等。
ReentrantReadWriteLock實現原理分析
- ReentrantReadWriteLock需要一個owner用來標記那個寫操作的線程獲取到了鎖,owner只會標記寫操作的線程引用,不會標記讀操作的線程,一個writeCount用來記錄寫操作加鎖的次數, 一個readCount用來記錄讀操作加鎖的次數,還有一個waiters等待隊列用來存放沒有搶到鎖的線程列表
- 當有寫操作線程進來時,會先判斷readCount的值,如果readCount為0說明讀鎖未被占用
- 然後判斷writeCount的值,如果writeCount為0,說明寫鎖未被占用
- 然後通過CAS操作進行搶鎖將writeCount值加1,如果搶到鎖則將owner設置為當前寫操作線程的引用
- 如果writeCount不為0同時owner指向當前寫線程的引用,則將writeCount的值加1
- 如果writeCount不為0同時owner指向的不是當前寫線程的引用,則將則將線程放入等待隊列
- 如果CAS搶鎖失敗,則將線程放入等待隊列
- 如果寫操作線程進來時,readCount不為0說明讀鎖已被占用,則將線程放入等待隊列
- 當有讀操作線程進來時,會先判斷writeCount的值,如果writeCount為0說明寫鎖未被占用
- 然後通過CAS將readCount的值加1
- 如果讀操作線程進來時,writeCount不為0說明寫鎖被占用
- 如果寫鎖是被當前線程占用則該線程可以繼續獲得讀鎖,即鎖降級
- 如果寫鎖不是被當前線程占用,則將線程放入等待隊列
- 當有寫線程釋放鎖時,會將writeCount的值減1,如果writeCount的值為0,則將owner設為null同時喚醒等待隊列頭部的線程出隊列進行搶鎖操作
- 如果等待隊列的頭部線程是讀操作,則會進行CAS操作將readCount值加1同時喚醒下一個等待線程
- 如果下一個線程還是讀操作,則會進行CAS操作將readCount值加1並且繼續喚醒下一個等待線程
- 如果下一個線程是寫操作,則不會喚醒需要等到將讀鎖釋放完之後才會喚醒
手動實現ReentrantReadWriteLock示例:
public class MyReadWriteLock {
private AtomicInteger readCount = new AtomicInteger(0);
private AtomicInteger writeCount = new AtomicInteger(0);
// 獨占鎖 擁有者
private AtomicReference<Thread> owner = new AtomicReference<>();
// 等待隊列
private volatile LinkedBlockingQueue<WaitNode> waiters = new LinkedBlockingQueue<WaitNode>();
class WaitNode {
int type = 0; // 0 為想獲取獨占鎖的線程, 1為想獲取共用鎖的線程
Thread thread = null;
int arg = 0;
public WaitNode(Thread thread, int type, int arg) {
this.thread = thread;
this.type = type;
this.arg = arg;
}
}
// 獲取獨占鎖
public void lockWrite() {
int arg = 1;
// 嘗試獲取獨占鎖,若成功,退出方法, 若失敗...
if (!tryLockWrite(arg)) {
// 標記為獨占鎖
WaitNode waitNode = new WaitNode(Thread.currentThread(), 0, arg);
waiters.offer(waitNode); // 進入等待隊列
// 迴圈嘗試拿鎖
for (; ; ) {
// 若隊列頭部是當前線程
WaitNode head = waiters.peek();
if (head != null && head.thread == Thread.currentThread()) {
if (!tryLockWrite(arg)) { // 再次嘗試獲取 獨占鎖
LockSupport.park(); // 若失敗,掛起線程
} else { // 若成功獲取
waiters.poll(); // 將當前線程從隊列頭部移除
return; // 並退出方法
}
} else { // 若不是隊列頭部元素
LockSupport.park(); // 將當前線程掛起
}
}
}
}
// 釋放獨占鎖
public boolean unlockWrite() {
int arg = 1;
// 嘗試釋放獨占鎖 若失敗返回true,若失敗...
if (tryUnlockWrite(arg)) {
WaitNode next = waiters.peek(); // 取出隊列頭部的元素
if (next != null) {
Thread th = next.thread;
LockSupport.unpark(th); // 喚醒隊列頭部的線程
}
return true; // 返回true
}
return false;
}
// 嘗試獲取獨占鎖
public boolean tryLockWrite(int acquires) {
// 如果read count !=0 返回false
if (readCount.get() != 0) return false;
int wct = writeCount.get(); // 拿到 獨占鎖 當前狀態
if (wct == 0) {
if (writeCount.compareAndSet(wct, wct + acquires)) { // 通過修改state來搶鎖
owner.set(Thread.currentThread()); // 搶到鎖後,直接修改owner為當前線程
return true;
}
} else if (owner.get() == Thread.currentThread()) {
writeCount.set(wct + acquires); // 修改count值
return true;
}
return false;
}
// 嘗試釋放獨占鎖
public boolean tryUnlockWrite(int releases) {
// 若當前線程沒有 持有獨占鎖
if (owner.get() != Thread.currentThread()) {
throw new IllegalMonitorStateException(); // 拋IllegalMonitorStateException
}
int wc = writeCount.get();
int nextc = wc - releases; // 計算 獨占鎖剩餘占用
writeCount.set(nextc); // 不管是否完全釋放,都更新count值
if (nextc == 0) { // 是否完全釋放
owner.compareAndSet(Thread.currentThread(), null);
return true;
} else {
return false;
}
}
// 獲取共用鎖
public void lockRead() {
int arg = 1;
if (tryLockRead(arg) < 0) { // 如果tryAcquireShare失敗
// 將當前進程放入隊列
WaitNode node = new WaitNode(Thread.currentThread(), 1, arg);
waiters.offer(node); // 加入隊列
for (; ; ) {
// 若隊列頭部的元素是當前線程
WaitNode head = waiters.peek();
if (head != null && head.thread == Thread.currentThread()) {
if (tryLockRead(arg) >= 0) { // 嘗試獲取共用鎖, 若成功
waiters.poll(); // 將當前線程從隊列中移除
WaitNode next = waiters.peek();
if (next != null && next.type == 1) { // 如果下一個線程也是等待共用鎖
LockSupport.unpark(next.thread); // 將其喚醒
}
return; // 退出方法
} else { // 若嘗試失敗
LockSupport.park(); // 掛起線程
}
} else { // 若不是頭部元素
LockSupport.park();
}
}
}
}
// 解鎖共用鎖
public boolean unLockRead() {
int arg = 1;
if (tryUnLockRead(arg)) { // 當read count變為0,才叫release share成功
WaitNode next = waiters.peek();
if (next != null) {
LockSupport.unpark(next.thread);
}
return true;
}
return false;
}
// 嘗試獲取共用鎖
public int tryLockRead(int acquires) {
for (; ; ) {
if (writeCount.get() != 0 && owner.get() != Thread.currentThread()) return -1;
int rct = readCount.get();
if (readCount.compareAndSet(rct, rct + acquires)) {
return 1;
}
}
}
// 嘗試解鎖共用鎖
public boolean tryUnLockRead(int releases) {
for (; ; ) {
int rc = readCount.get();
int nextc = rc - releases;
if (readCount.compareAndSet(rc, nextc)) {
return nextc == 0;
}
}
}
}
鎖降級
鎖降級指的是寫鎖降級為讀鎖,是指持有寫鎖的同時,再獲取讀鎖,隨後釋放寫鎖的過程。
寫鎖是線程獨占,讀鎖是線程共用,所以寫鎖降級為讀鎖可行,而讀鎖升級為寫鎖不可行。
代碼示例:
class TeacherInfoCache {
static volatile boolean cacheValid;
static final ReadWriteLock rwl = new ReentrantReadWriteLock();
static Object get(String dataKey) {
Object data = null;
// 讀取數據,加讀鎖
rwl.readLock().lock();
try {
if (cacheValid) {
data = Redis.data.get(dataKey);
} else {
// 通過加鎖的方式去訪問DB,加寫鎖
rwl.readLock().unlock();
rwl.writeLock().lock();
try {
if (!cacheValid) {
data = DataBase.queryUserInfo();
Redis.data.put(dataKey, data);
cacheValid = true;
}
} finally {
// 鎖降級
rwl.readLock().lock();
rwl.writeLock().unlock();
}
}
return data;
} finally {
rwl.readLock().unlock();
}
}
}
class DataBase {
static String queryUserInfo() {
System.out.println("查詢資料庫。。。");
return "name:Kody,age:40,gender:true,";
}
}
class Redis {
static Map<String, Object> data = new HashMap<>();
}
更多相關文章
-
Java8 增加了 Lambda 表達式,很大程度使代碼變的更加簡潔緊湊了,那麼 Java8 是如何實現 Lambda 表達式的呢? 直接看一個簡單的創建線程的例子。 執行 編譯生成文件 ,然後用 命令來分析這個class文件。 執行 顯示所有類和成員。 由上面的代碼可以看出編譯器根據 Lambda ...
-
一、原型模式簡介 1、基礎概念 原型模式屬於對象的創建模式。通過給出一個原型對象來指明所有創建的對象的類型,然後用複製這個原型對象的辦法創建出更多同類型的對象。 2、模式結構 原型模式要求對象實現一個可以“克隆”自身的介面,這樣就可以通過複製一個實例對象本身來創建一個新的實例。這樣一來,通過原型實例 ...
-
https://www.cnblogs.com/fireflyupup/p/4875130.html Collection List 在Collection的基礎上引入了有序的概念,位置精確;允許相同元素。在列表上迭代通常優於索引遍歷。特殊的ListIterator迭代器允許元素插入、替換,雙向訪問 ...
-
一個可以沉迷於技術的程式猿,wx加入加入技術群:fsx641385712 ...
-
1.字元串的定義 可以使用""雙引號,也可以使用''單引號定義字元串,一般使用雙引號定義。 2.字元串的操作 判斷類型: 查找和替換 大小寫切換: 文本對齊 註:string.center(weight,str) 以str填充對齊,其他兩個方法類似,都可以拓展。 去除空白字元 拆分和鏈接 3.字元串 ...
-
新聞 "Fantomas 3.0" "宣告.NET Core 3.0預覽版7" ".NET Core 3.0預覽版7中ASP.NET Core與Blazor的升級" "Visual Studio 2019版本16.2正式版本與16.3預覽版1" "Mac上的Visual Studio 2019版本8 ...
-
1.數組: java.lang.ArrayIndexOutOfBoundsException:5 下標越界異常 java.lang.NullPointerException 空指針異常 arr.length獲取數組長度 數組存儲的是多個數,數據的操作離不開迴圈2數組初始化:int[] arr=new ...
-
一、Gateway 和 Zuul 的區別 Zuul 基於servlet 2.5 (works with 3.x),使用阻塞API。它不支持任何長期的連接,如websocket。 Gateway建立在Spring Framework 5,Project Reactor 和Spring Boot 2 上 ...