聊聊Java三種常見的分布式鎖
平時我們在 APP 上搶某件商品時,手指瘋狂點擊下單,但是只會生成一個訂單,為什么呢?
因為對用戶下單加了一個鎖,避免用戶重復(fù)下單。而在分布式場景,訂單可以看作一個共享資源,所以這個鎖可以叫做分布式鎖。
一般來講,分布式鎖有三種實現(xiàn)方式:
- 基于數(shù)據(jù)庫的分布式鎖
- 基于 redis 的分布式鎖
- 基于 Zookeeper 的分布式鎖
1 基于數(shù)據(jù)庫的分布式鎖
基于數(shù)據(jù)庫實現(xiàn)分布式鎖有很多實現(xiàn),這里介紹其中一種。其核心思想是:
在數(shù)據(jù)庫中創(chuàng)建一個表,表中添加 方法名 等字段,并 對方法名字段添加唯一索引,如果要對某個方法加鎖,則使用這個方法名向表中插入數(shù)據(jù),成功插入則加鎖成功,方法執(zhí)行完后刪除對應(yīng)的行數(shù)據(jù)釋放鎖。[1]
1.1 創(chuàng)建表
CREATE TABLE `t_database_lock` ( `id` bigint NOT NULL COMMENT '主鍵', `method_name` varchar(50) COLLATE utf8_bin NOT NULL DEFAULT '方法名字', `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '創(chuàng)建時間', `update_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '最近更新時間' PRIMARY KEY(`id`), UNIQUE INDEX `uk_t_database_lock_mname`(`method_name`) USING BTREE COMMENT '方法名唯一索引', INDEX `idx_t_accounting_balance_account_utime`(`update_time`) USING BTREE COMMENT '更新時間索引', INDEX `idx_t_accounting_balance_account_ctime`(`create_time`) USING BTREE COMMENT '創(chuàng)建時間索引' ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin
1.2 偽代碼實現(xiàn)
主流程如下:
public void execute(String methodName) {
try {
if (lock(methodName)) { // 加鎖
// 你的業(yè)務(wù)邏輯
run();
}
} finally {
unlock(); // 釋放鎖
}
}lock() 方法的實現(xiàn)的偽代碼如下:
private boolean lock(String methodName) {
try {
// 插入一條數(shù)據(jù)
int count = "INSERT INTO method_lock (method_name, desc) VALUES ('methodName', '需要加鎖的methodName');";
if (count == 1) { // 1 表示插入成功
return true;
}
return false;
} catch (Exception e) {
return false;
}
return false;
}unlock() 方法實現(xiàn)的偽代碼如下:
private boolean lock(String methodName) {
try {
// 插入一條數(shù)據(jù)
int count = "INSERT INTO method_lock (method_name, desc) VALUES ('methodName', '需要加鎖的methodName');";
if (count == 1) { // 1 表示插入成功
return true;
}
return false;
} catch (Exception e) {
return false;
}
return false;
}它的優(yōu)點是實現(xiàn)簡單,但是有以下缺點:
- 讀取寫入性能低,資源開銷較大,不適合并發(fā)量很高的場景。
- 可靠性較低,因為數(shù)據(jù)庫的可用性會直接影響到分布式鎖的可用性,所以數(shù)據(jù)庫需要雙機部署、數(shù)據(jù)同步、主備切換。
- 沒有鎖失效機制。如果在成功插入一條數(shù)據(jù)后,服務(wù)器宕機了,導(dǎo)致這條數(shù)據(jù)沒有刪除,服務(wù)恢復(fù)后一直獲取不到鎖。解決方式是可以在表中新增一例失效時間,定時清除這些失效數(shù)據(jù)。
- 鎖邏輯簡單,不具備阻塞鎖等高級功能。
2 基于 redis 的分布式鎖
redis 分布式鎖的好處是性能好,實現(xiàn)也較為方便。
2.1 setnx + expire
- setnx 即 Set if Not Exists,如果 redis 存在這個 key,setnx 返回0,加鎖失??;否則返回1,加鎖成功。
- expire 用來設(shè)置 redis key 的過期時間,避免這個鎖一直存在,造成無法加鎖的問題。
實現(xiàn)偽代碼如下:
if (jedis.setnx(key) == 1) { // 加鎖成功
expire(key, 100);
try {
doSomething(); //業(yè)務(wù)處理
} catch (Exception e) {
} finally {
jedis.del(key); // 釋放鎖
}
}這個是基本的實現(xiàn),但是會有一個問題:如果 setnx 執(zhí)行結(jié)束,進程中斷了,沒有執(zhí)行 expire 代碼,導(dǎo)致 key 沒有設(shè)置過期時間,別的線程就永遠(yuǎn)獲取不到這個鎖了。
2.2 setnx 同時設(shè)置過期時間
基于以上問題,我們可以將 redis 的 value 利用起來,將 jedis.setnx(key) 改為 jedis.setnx(key, ${過期時間}),加鎖的偽代碼如下[2]:
public boolean lock(String key, Long expireTime) {
Jedis jedis = new Jedis();
//系統(tǒng)時間 + 設(shè)置的過期時間
long expires = System.currentTimeMillis() + expireTime;
String expiresStr = String.valueOf(expires);
// 如果當(dāng)前鎖不存在,返回加鎖成功
if (jedis.setnx(key, expiresStr) == 1) {
return true;
}
// 如果鎖已經(jīng)存在,獲取鎖的過期時間
String currentValueStr = jedis.get(key);
// 鎖過期
if (currentValueStr != null && Long.parseLong(currentValueStr) < System.currentTimeMillis()) {
// 鎖已過期,獲取上一個鎖的過期時間,并設(shè)置現(xiàn)在鎖的過期時間
String oldValueStr = jedis.getSet(key, expiresStr);
if (oldValueStr != null && oldValueStr.equals(currentValueStr)) {
// 只有上一個鎖的過期時間和當(dāng)前過期時間相同才可加鎖
return true;
}
}
return false;
}以上的實現(xiàn)方式避免了鎖一直存在的問題,但是仍然有缺點:
- 過期時間使用
System.currentTimeMillis()生成,意味著客戶端時間必須同步。 - 沒有辨別客戶端的機制,可能 A 機器加了鎖,被 B 機器解鎖了。
- 并發(fā)情況下,多個機器同時加鎖,只有一個會加鎖成功。
2.3 將 redis value 設(shè)置為客戶端標(biāo)識
通過以上分析,我們既希望 redis 設(shè)置 key 和 expire 同時進行,又希望 redis 加鎖解鎖時能識別客戶端。
因此我們將客戶端加鎖時,生成一個唯一 ID 作為 redis 的 value,在進行鎖操作時先校驗一下客戶端是否一致,在進行鎖操作。redis 的 set() 方法可以滿足要求。
流程的偽代碼如下:
public void execute() {
Jedis jedis = new Jedis();
String key = "key";
String uuid = "uuid";
if ("1".equals(jedis.set(key, uuid, "NX", "EX", 100))) { //加鎖
try {
doSomething(); //業(yè)務(wù)處理
} finally {
//判斷是不是當(dāng)前線程加的鎖,是才釋放
if (uuid.equals(jedis.get(key))) {
jedis.del(key); //釋放鎖
}
}
}
}以上實現(xiàn)已經(jīng)不錯了,可以滿足大部分場景。但是還有一個問題:過期時間是個不確定因素,你很難設(shè)置一個合適的過期時間,使得鎖既在業(yè)務(wù)執(zhí)行后釋放,又不至于太長導(dǎo)致其他線程獲取失敗。
2.4 Redisson
Redisson 可以解決上面的問題,它的原理是在加鎖成功后啟動一個看門狗,在線程持有鎖期間不斷延長 key 的生存時間,如此一來可解決鎖在業(yè)務(wù)執(zhí)行完之前就釋放的問題。
3 Zookeeper 分布式鎖
當(dāng)使用 ZooKeeper 實現(xiàn)分布式鎖時,可以按照以下步驟進行操作:
- 每個客戶端連接到 ZooKeeper 服務(wù)器。
- 每個客戶端在 ZooKeeper 上創(chuàng)建一個臨時順序節(jié)點作為鎖節(jié)點。
- 客戶端獲取鎖的方式是判斷自己創(chuàng)建的節(jié)點是否是當(dāng)前所有鎖節(jié)點中最小的。
- 如果客戶端創(chuàng)建的節(jié)點是最小節(jié)點,表示該客戶端獲得了鎖,可以執(zhí)行關(guān)鍵區(qū)域的代碼。
- 如果客戶端創(chuàng)建的節(jié)點不是最小節(jié)點,則需要監(jiān)聽比自己創(chuàng)建節(jié)點次小的節(jié)點。
- 當(dāng)監(jiān)聽到次小節(jié)點被刪除時,客戶端重新檢查自己創(chuàng)建的節(jié)點是否是當(dāng)前所有鎖節(jié)點中最小的。如果是,則獲得了鎖,可以執(zhí)行關(guān)鍵區(qū)域的代碼。
- 當(dāng)客戶端完成了關(guān)鍵區(qū)域的操作后,釋放鎖,即刪除自己創(chuàng)建的節(jié)點。
這樣,通過 ZooKeeper 的臨時順序節(jié)點和監(jiān)聽機制,可以實現(xiàn)一個簡單的分布式鎖。每個客戶端在創(chuàng)建節(jié)點時,會根據(jù)節(jié)點名稱的順序來確定是否獲得了鎖。如果創(chuàng)建的節(jié)點是最小節(jié)點,則表示該客戶端獲得了鎖,可以執(zhí)行關(guān)鍵區(qū)域的代碼。其他客戶端則通過監(jiān)聽比自己創(chuàng)建的節(jié)點次小的節(jié)點來等待鎖的釋放,從而實現(xiàn)分布式的互斥訪問。
總結(jié)
三種分布式鎖對比
數(shù)據(jù)庫分布式鎖實現(xiàn)
優(yōu)點:簡單,使用方便,不需要引入 Redis、Zookeeper 等中間件。
缺點:
不適合高并發(fā)的場景
db 操作性能較差
Redis 分布式鎖實現(xiàn)
優(yōu)點:
性能好,適合高并發(fā)場景
較輕量級
有較好的框架支持,如 Redisson
缺點:
過期時間不好控制
需要考慮鎖被別的線程誤刪場景
Zookeeper 分布式鎖實現(xiàn)
缺點:
性能不如 Redis 實現(xiàn)的分布式鎖
比較重的分布式鎖。
優(yōu)點:
有較好的性能和可靠性
有封裝較好的框架,如 Curator
以上實現(xiàn)方式都不是完美的,在實際生產(chǎn)中要合理評估,選擇適合自己的方案。
性能
redis > zookeeper >= 數(shù)據(jù)庫
可靠性
zookeeper > redis > 數(shù)據(jù)庫
實現(xiàn)復(fù)雜度
zookeeper > redis > 數(shù)據(jù)庫
到此這篇關(guān)于聊聊Java三種常見的分布式鎖的文章就介紹到這了,更多相關(guān)Java 分布式鎖內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Mybatis通過數(shù)據(jù)庫表自動生成實體類和xml映射文件
這篇文章主要介紹了Mybatis通過數(shù)據(jù)庫表自動生成實體類和xml映射文件的操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07
使用Java實現(xiàn)動態(tài)生成MySQL數(shù)據(jù)庫
這篇文章主要為大家詳細(xì)介紹了如何使用Java實現(xiàn)動態(tài)生成MySQL數(shù)據(jù)庫,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-02-02
使用Homebrew配置Java開發(fā)環(huán)境操作方法
下面小編就為大家?guī)硪黄褂肏omebrew配置Java開發(fā)環(huán)境操作方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-06-06

