国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務器之家:專注于服務器技術及軟件下載分享
分類導航

Mysql|Sql Server|Oracle|Redis|MongoDB|PostgreSQL|Sqlite|DB2|mariadb|Access|數據庫技術|

服務器之家 - 數據庫 - Redis - Redis實現分布式鎖的幾種方案

Redis實現分布式鎖的幾種方案

2023-05-07 05:10未知服務器之家 Redis

1.前言 對于Redis實現分布式鎖的幾種方案這個話題,展開之前我想先簡單聊聊什么是分布式鎖,分布式鎖的使用場景,除了Redis外還有什么技術實現分布式鎖等一系列內容。 1.1分布式鎖 說大一點,就是在現在發展越來越迅速的大背

1.前言

對于Redis實現分布式鎖的幾種方案這個話題,展開之前我想先簡單聊聊什么是分布式鎖,分布式鎖的使用場景,除了Redis外還有什么技術實現分布式鎖等一系列內容。

1.1分布式鎖

說大一點,就是在現在發展越來越迅速的大背景下,去中心化分布式系統越來越普及,在我們實際的生產開發當中,有一種不可避免的場景就是多個進程互斥的對其資源的使用,為了保證數據不重復,要求在同一時刻,同一任務只在一個節點上運行,且保證在多進程下的數據安全,分布式鎖就十分重要了。

1.2分布式鎖的幾種方案

方式有很多種,根據技術角度的不同

有基于MySQL的方式,通過表的唯一索引,通過insert和delete就可以實現加鎖和解鎖的效果;

有基于zookeeper的方式,通過創建臨時有序節點,判斷創建的節點序號是否最小。若是,則表示獲取到鎖,不是,則watch /lock目錄下序號比自身小的前一個節點,解鎖只需要刪除節點;

有基于Redis的方式。通過執行setnx,若成功再執行expire添加過期時間的方式加鎖,解鎖執行delete命令。

方式有很多,不一一列舉了。

1.3Redis分布式鎖需要滿足的條件

  • 互斥性。在任意時刻,只有一個客戶端能持有鎖。
  • 不發生死鎖。即使有一個客戶端在持有鎖的期間崩潰而沒有主動解鎖也能保證后續其他客戶端能加鎖。
  • 同一性。加鎖和解鎖必須是同一個客戶端,客戶端自己不能把別人加的鎖給解了,即不能誤解鎖。
  • 容錯性。只要大多數Redis節點正常運行,客戶端就能夠獲取和釋放鎖。

2.Redis實現分布式鎖的幾種方案

可以通過以下方式實現(包括但不限于):

  • SETNX + EXPIRE
  • SETNX + value(系統時間+過期時間)
  • 通過開源框架-Redisson

簡單來說說,用Java代碼演示:

2.1 SETNX + EXPIRE

setnx(SET IF NOT EXISTS)+ expire命令。先用setnx來搶鎖,如果搶到鎖,再用expire給鎖設置一個過期時間,這樣持有鎖超時時釋放鎖,防止鎖忘記釋放。但此時setnx和expire兩個命令無法保證原子性,例如:

if(jedis.setnx(key_resource_id,lock_value) == 1){ //加鎖
expire(key_resource_id,100); //設置過期時間
try {
//業務代碼塊
}catch() {
}finally {
jedis.del(key_resource_id); //釋放鎖
}
}

2.2 SETNX + value(系統時間+過期時間)

可以把過期時間放到setnx的value值里面。如果加鎖失敗,再拿出value值校驗一下即可。加鎖代碼如下:

long expires = System.currentTimeMillis() + expireTime; //系統時間+設置的過期時間
String expiresStr = String.valueOf(expires);
// 如果當前鎖不存在,則加鎖成功
if (jedis.setnx(key_resource_id, expiresStr) == 1) {
return true;
}
// 如果鎖已經存在,獲取鎖的過期時間
String currentValueStr = jedis.get(key_resource_id);

// 如果獲取到的過期時間,小于系統當前時間,表示已經過期
if (currentValueStr != null && Long.parseLong(currentValueStr) < System.currentTimeMillis()) {
// 鎖已過期,獲取上一個鎖的過期時間,并設置現在鎖的過期時間
String oldValueStr = jedis.getSet(key_resource_id, expiresStr);
if (oldValueStr != null && oldValueStr.equals(currentValueStr)) {
// 考慮多線程并發的情況,只有一個線程的設置值和當前值相同,它才可以加鎖
return true;
}
}
//其他情況均返回加鎖失敗
return false;

2.3 通過開源框架-Redisson

那么此時就要去想了,如果已經超過了加鎖的過期時間,可是業務還沒執行完成,這個時候怎么做呢?是把過期時間延長嗎?顯然不合理,可以通過開源框架-Redisson優化這個問題,簡單來說,Redisson就是當一個線程獲得鎖以后,給該線程開啟一個定時守護線程,每隔一段時間檢查鎖是否還存在,存在則對鎖的過期時間延長,防止鎖過期提前釋放。假設兩個線程爭奪統一公共資源:線程A獲取鎖,并通過哈希算法選擇節點,執行Lua腳本加鎖,同時其看門狗機制會啟動一個watch dog(后臺線程),每隔10秒檢查線程,如果線程A還持有鎖,那么就會不斷的延長鎖key的生存時間。線程B獲得鎖失敗,就會訂閱解鎖消息,當獲取鎖到剩余過期時間后,調用信號量方法阻塞住,直到被喚醒或等待超時。一旦線程A釋放了鎖,就會廣播解鎖消息。于是,解鎖消息的監聽器會釋放信號量,獲取鎖被阻塞的線程B就會被喚醒,并重新嘗試獲取鎖。

Redisson 支持單點模式、主從模式、哨兵模式、集群模式,假設現為單點模式:

//構造Config
Config config = new Config();
config.useSingleServer().setAddress("redis://ip:port").setPassword("Password.~#").setDatabase(0);
//構造RedissonClient
RedissonClient redissonClient = Redisson.create(config);
//獲取鎖實例
RLock rLock = redissonClient.getLock(lockKey);
try {
//獲取鎖,waitTimeout為最大等待時間,超過這個值,則認為獲取鎖失敗。leaseTime為鎖的持有時間
boolean res = rLock.tryLock((long)waitTimeout, (long)leaseTime, TimeUnit.SECONDS);
if (res) {
//業務塊
}
} catch (Exception e) {
}finally{
//解鎖
rLock.unlock();
}

3.小結

Redis的分布式鎖實現方式有很多,這里不一一列舉了,有機會再展開Lua腳本、分布式鎖Redlock等內容。

延伸 · 閱讀

精彩推薦
  • Redisredis 交集、并集、差集的具體使用

    redis 交集、并集、差集的具體使用

    這篇文章主要介紹了redis 交集、并集、差集的具體使用,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友...

    xiaojin21cen10152021-07-27
  • RedisRedis如何實現數據庫讀寫分離詳解

    Redis如何實現數據庫讀寫分離詳解

    Redis的主從架構,能幫助我們實現讀多,寫少的情況,下面這篇文章主要給大家介紹了關于Redis如何實現數據庫讀寫分離的相關資料,文中通過示例代碼介紹...

    羅兵漂流記6092019-11-11
  • Redisredis中如何使用lua腳本讓你的靈活性提高5個逼格詳解

    redis中如何使用lua腳本讓你的靈活性提高5個逼格詳解

    這篇文章主要給大家介紹了關于redis中如何使用lua腳本讓你的靈活性提高5個逼格的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具...

    一線碼農5812019-11-18
  • Redisredis實現排行榜功能

    redis實現排行榜功能

    排行榜在很多地方都能使用到,redis的zset可以很方便地用來實現排行榜功能,本文就來簡單的介紹一下如何使用,具有一定的參考價值,感興趣的小伙伴們...

    乘月歸5022021-08-05
  • RedisRedis 事務知識點相關總結

    Redis 事務知識點相關總結

    這篇文章主要介紹了Redis 事務相關總結,幫助大家更好的理解和學習使用Redis,感興趣的朋友可以了解下...

    AsiaYe8232021-07-28
  • RedisRedis全量復制與部分復制示例詳解

    Redis全量復制與部分復制示例詳解

    這篇文章主要給大家介紹了關于Redis全量復制與部分復制的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用Redis爬蟲具有一定的參考學習...

    豆子先生5052019-11-27
  • RedisRedis的配置、啟動、操作和關閉方法

    Redis的配置、啟動、操作和關閉方法

    今天小編就為大家分享一篇Redis的配置、啟動、操作和關閉方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧 ...

    大道化簡5312019-11-14
  • Redis詳解Redis復制原理

    詳解Redis復制原理

    與大多數db一樣,Redis也提供了復制機制,以滿足故障恢復和負載均衡等需求。復制也是Redis高可用的基礎,哨兵和集群都是建立在復制基礎上實現高可用的...

    李留廣10222021-08-09
主站蜘蛛池模板: 国产91色| 欧美一区二区在线观看 | av在线免费观看网址 | 中文在线a在线 | 成人瑟瑟 | 国产黄色电影 | 欧美日韩精品一区二区三区四区 | 成人精品久久久 | 波多一区二区 | 日韩中文字幕在线视频 | 欧美亚洲高清 | 亚洲激情视频 | 极品美女销魂一区二区三区 | 日本一区高清 | 内地农村三片在线观看 | 国产欧美精品区一区二区三区 | 亚洲欧洲成人 | 免费观看aaa | 欧美麻豆视频 | 亚洲一区二区三区在线 | 在线免费日韩 | 欧美日韩高清 | 天天干天天干天天干天天射 | 日韩国产一区二区 | 国产中文字幕一区 | 国产视频一二区 | 日韩中文字幕在线观看视频 | 欧美久久免费 | 一区二区三区四区在线 | 欧美一级视频 | 国产一级片儿 | 色678黄网站全部免费 | 色天天综合 | 午夜精品久久久久久久男人的天堂 | 午夜电影网 | 中文字幕在线观看视频地址二 | 国产精品美女久久久久久免费 | 一区二区三区在线免费观看 | 97视频免费在线观看 | 久久精品a一级国产免视看成人 | 伊人狠狠|