<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
說明
本文介紹為什麼要使用Redis的紅鎖(Redlock)、什麼是Redis的紅鎖以及Redis紅鎖的原理。
本文用Redisson來介紹Redis紅鎖的用法。
Redisson 高版本會根據redisClient的模式來決定getLock返回的鎖型別,如果叢集模式,滿足紅鎖的條件,則會直接返回紅鎖。
官網
REDIS distlock -- Redis中國使用者組(CRUG)
主從結構分散式鎖的問題
實現Redis分散式鎖的最簡單的方法就是在Redis中建立一個key,這個key有一個失效時間(TTL),以保證鎖最終會被自動釋放掉。當用戶端釋放資源(解鎖)的時候,會刪除掉這個key。
從表面上看似乎效果不錯,但有一個嚴重的單點失敗問題:如果Redis掛了怎麼辦?你可能會說,可以通過增加一個slave節點解決這個問題。但這通常是行不通的。這樣做,我們不能實現資源的獨享,因為Redis的主從同步通常是非同步的。
在這種場景(主從結構)中存在明顯的競態:
有時候程式就是這麼巧,比如說正好一個節點掛掉的時候,多個使用者端同時取到了鎖。如果你可以接受這種小概率錯誤,那用這個基於複製的方案就完全沒有問題。否則的話,我們建議你實現下面描述的解決方案。
簡介
Redis中針對此種情況,引入了紅鎖的概念。紅鎖採用主節點過半機制,即獲取鎖或者釋放鎖成功的標誌為:在過半的節點上操作成功。
原理
在Redis的分散式環境中,我們假設有N個Redis master。這些節點完全互相獨立,不存在主從複製或者其他叢集協調機制。之前我們已經描述了在Redis單範例下怎麼安全地獲取和釋放鎖。我們確保將在每(N)個範例上使用此方法獲取和釋放鎖。在這個樣例中,我們假設有5個Redis master節點,這是一個比較合理的設定,所以我們需要在5臺機器上面或者5臺虛擬機器器上面執行這些範例,這樣保證他們不會同時都宕掉。
為了取到鎖,使用者端應該執行以下操作:
官網
官方github:8. 分散式鎖和同步器 · redisson/redisson Wik
基於Redis的Redisson紅鎖RedissonRedLock物件實現了Redlock介紹的加鎖演演算法。該物件也可以用來將多個RLock物件關聯為一個紅鎖,每個RLock物件範例可以來自於不同的Redisson範例。
RLock lock1 = redissonInstance1.getLock("lock1"); RLock lock2 = redissonInstance2.getLock("lock2"); RLock lock3 = redissonInstance3.getLock("lock3"); RedissonRedLock lock = new RedissonRedLock(lock1, lock2, lock3); // 同時加鎖:lock1 lock2 lock3 // 紅鎖在大部分節點上加鎖成功就算成功。 lock.lock(); ... lock.unlock();
大家都知道,如果負責儲存某些分散式鎖的某些Redis節點宕機以後,而且這些鎖正好處於鎖住的狀態時,這些鎖會出現鎖死的狀態。為了避免這種情況的發生,Redisson內部提供了一個監控鎖的看門狗,它的作用是在Redisson範例被關閉前,不斷的延長鎖的有效期。預設情況下,看門狗的檢查鎖的超時時間是30秒鐘,也可以通過修改Config.lockWatchdogTimeout來另行指定。
另外Redisson還通過加鎖的方法提供了leaseTime的引數來指定加鎖的時間。超過這個時間後鎖便自動解開了。
RedissonRedLock lock = new RedissonRedLock(lock1, lock2, lock3); // 給lock1,lock2,lock3加鎖,如果沒有手動解開的話,10秒鐘後將會自動解開 lock.lock(10, TimeUnit.SECONDS); // 為加鎖等待100秒時間,並在加鎖成功10秒鐘後自動解開 boolean res = lock.tryLock(100, 10, TimeUnit.SECONDS); ... lock.unlock();
RedissonRedLock extends RedissonMultiLock,所以實際上,redLock.tryLock實際呼叫:org.redisson.RedissonMultiLock.java#tryLock(),進而呼叫到其同類的tryLock(long waitTime, long leaseTime, TimeUnit unit) ,入參為:tryLock(-1, -1, null)
org.redisson.RedissonMultiLock.java#tryLock(long waitTime, long leaseTime, TimeUnit unit)原始碼如下:
final List<RLock> locks = new ArrayList<>(); /** * Creates instance with multiple {@link RLock} objects. * Each RLock object could be created by own Redisson instance. * * @param locks - array of locks */ public RedissonMultiLock(RLock... locks) { if (locks.length == 0) { throw new IllegalArgumentException("Lock objects are not defined"); } this.locks.addAll(Arrays.asList(locks)); } public boolean tryLock(long waitTime, long leaseTime, TimeUnit unit) throws InterruptedException { long newLeaseTime = -1; if (leaseTime != -1) { newLeaseTime = unit.toMillis(waitTime)*2; } long time = System.currentTimeMillis(); long remainTime = -1; if (waitTime != -1) { remainTime = unit.toMillis(waitTime); } long lockWaitTime = calcLockWaitTime(remainTime); /** * 1. 允許加鎖失敗節點個數限制(N-(N/2+1)) */ int failedLocksLimit = failedLocksLimit(); /** * 2. 遍歷所有節點通過EVAL命令執行lua加鎖 */ List<RLock> acquiredLocks = new ArrayList<>(locks.size()); for (ListIterator<RLock> iterator = locks.listIterator(); iterator.hasNext();) { RLock lock = iterator.next(); boolean lockAcquired; /** * 3.對節點嘗試加鎖 */ try { if (waitTime == -1 && leaseTime == -1) { lockAcquired = lock.tryLock(); } else { long awaitTime = Math.min(lockWaitTime, remainTime); lockAcquired = lock.tryLock(awaitTime, newLeaseTime, TimeUnit.MILLISECONDS); } } catch (RedisResponseTimeoutException e) { // 如果丟擲這類異常,為了防止加鎖成功,但是響應失敗,需要解鎖所有節點 unlockInner(Arrays.asList(lock)); lockAcquired = false; } catch (Exception e) { // 丟擲異常表示獲取鎖失敗 lockAcquired = false; } if (lockAcquired) { /** *4. 如果獲取到鎖則新增到已獲取鎖集合中 */ acquiredLocks.add(lock); } else { /** * 5. 計算已經申請鎖失敗的節點是否已經到達 允許加鎖失敗節點個數限制 (N-(N/2+1)) * 如果已經到達, 就認定最終申請鎖失敗,則沒有必要繼續從後面的節點申請了 * 因為 Redlock 演演算法要求至少N/2+1 個節點都加鎖成功,才算最終的鎖申請成功 */ if (locks.size() - acquiredLocks.size() == failedLocksLimit()) { break; } if (failedLocksLimit == 0) { unlockInner(acquiredLocks); if (waitTime == -1 && leaseTime == -1) { return false; } failedLocksLimit = failedLocksLimit(); acquiredLocks.clear(); // reset iterator while (iterator.hasPrevious()) { iterator.previous(); } } else { failedLocksLimit--; } } /** * 6.計算 目前從各個節點獲取鎖已經消耗的總時間,如果已經等於最大等待時間,則認定最終申請鎖失敗,返回false */ if (remainTime != -1) { remainTime -= System.currentTimeMillis() - time; time = System.currentTimeMillis(); if (remainTime <= 0) { unlockInner(acquiredLocks); return false; } } } if (leaseTime != -1) { List<RFuture<Boolean>> futures = new ArrayList<>(acquiredLocks.size()); for (RLock rLock : acquiredLocks) { RFuture<Boolean> future = ((RedissonLock) rLock).expireAsync(unit.toMillis(leaseTime), TimeUnit.MILLISECONDS); futures.add(future); } for (RFuture<Boolean> rFuture : futures) { rFuture.syncUninterruptibly(); } } /** * 7.如果邏輯正常執行完則認為最終申請鎖成功,返回true */ return true; }
到此這篇關於Redis中Redisson紅鎖(Redlock)使用原理的文章就介紹到這了,更多相關Redis Redisson紅鎖內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45