手写分布式锁
qa
data:image/s3,"s3://crabby-images/d0184/d0184c72f27ca2c9537563a6f83434ac3ab8a98c" alt=""
redis除了做缓存,还有什么用法
data:image/s3,"s3://crabby-images/fbc52/fbc52a43cbd1ae17bb30bec4d8c3480cd72741bc" alt=""
data:image/s3,"s3://crabby-images/702b9/702b93fe2bd19f9b89325c7e93d7836270c20167" alt=""
redis 单机与集群的cap分析
data:image/s3,"s3://crabby-images/6eec7/6eec77facdd8d670fe1831e5eda5655371306e4b" alt=""
锁的种类data:image/s3,"s3://crabby-images/5a4c5/5a4c5c748b253a717ecdbe8dc33c65d747d8af34" alt=""
一个分布式锁需要满足的条件和刚需
- 独占性:任何时间只能有一个线程占有
- 高可用:
- 在redis集群环境下,不能因为一个节点挂了而出现获取锁和释放锁失败的情况
- 高并发请求下,依旧性能 ok
- 防死锁:杜绝死锁,必须有超时控制机制与撤销操作,有个兜底终止跳出方案
- 不乱抢:防止张冠李戴,不能私下unlock别人的锁,只能自己加锁自己释放,自己约的锁含着泪也要自己解
- 重入性:同一个节点的同一个线程,获得锁之后,他也可以再次获取这个锁
分布式锁及其重点
data:image/s3,"s3://crabby-images/412ed/412ed520115a2a84c5d478b8a096b3a3a49bcae7" alt=""
data:image/s3,"s3://crabby-images/8b0bb/8b0bbc2d3f6082f80e5bebc737da084c538454bd" alt=""
setnx 来获取锁
data:image/s3,"s3://crabby-images/56541/56541884df4bff18a30098212a83d5792e985762" alt=""
boot+redis 基础案例
使用场景
data:image/s3,"s3://crabby-images/a6e11/a6e11d88efde1a3524cc6d0ced05a4f5f5224839" alt=""
java
@Service
@Slf4j
public class InventoryService {
@Autowired
private StringRedisTemplate stringRedisTemplate;
@Value("${server.port}")
private String port;
private Lock lock = new ReentrantLock();
public String sale() {
String retMessage = "";
lock.lock();
try {
//1 查询库存信息
String result = stringRedisTemplate.opsForValue().get("inventory001");
//2 判断库存是否足够
Integer inventoryNumber = result == null? 0 : Integer.parseInt(result);
//3 扣除库存,每次减少一个
if (inventoryNumber > 0) {
inventoryNumber--;
stringRedisTemplate.opsForValue().set("inventory001", String.valueOf(inventoryNumber));
retMessage = "成功卖出一个商品,剩余库存" + inventoryNumber;
System.out.println(retMessage + ",端口号:" + port);
}else {
retMessage = "商品卖完了";
}
} finally {
lock.unlock();
}
return retMessage;
}
}
java
@RestController
@Api(tags = "redis 分布式锁测试")
public class InventoryController {
@Autowired
private InventoryService inventoryService;
@ApiOperation("扣除库存,一次卖一个")
@GetMapping("/inventory/sale")
public String sale() {
return inventoryService.sale();
}
}
GET http://localhost:6000/inventory/sale
手写分布式锁分析
data:image/s3,"s3://crabby-images/3a62f/3a62ffa2cd05b1f40ac3fb60bba455e0833c4039" alt=""
data:image/s3,"s3://crabby-images/a20a7/a20a75bd71531f54d04c791d1895d41f4738f732" alt=""
拷贝出来一份
nginx 负载均衡
nginx
upstream stock {
server 127.0.0.1:6001 weight=1;
server 127.0.0.1:6000 weight=1;
}
server {
listen 9000;
server_name localhost;
location / {
proxy_pass http://stock;
index index.html index.htm;
}
}
http
GET http://localhost:9000/inventory/sale
jmeter
添加线程组 100*1
data:image/s3,"s3://crabby-images/1fcb8/1fcb8f3012d8e41696fc27b289f04c7bd5ac6fa3" alt=""
添加取样器-http请求
data:image/s3,"s3://crabby-images/295ed/295ed7a68696699efb66f8ba45d93bfbfff1ba4c" alt=""
添加 listener-聚合报告
data:image/s3,"s3://crabby-images/ac25f/ac25f259906104942c0cd010084c7420077c1e82" alt=""
jmeter 100个线程1s执行完毕
执行完毕查看 库存数
data:image/s3,"s3://crabby-images/7ee19/7ee19085bf84cd359a9964c88b61e9f2ea220532" alt=""
为什么加了synchronized 和 lock 但是没有控制住
data:image/s3,"s3://crabby-images/6cc21/6cc21d7e686ceebcd5db8c42cd8e333c0d582b97" alt=""
data:image/s3,"s3://crabby-images/eeade/eeade5f13a72fcb5f34158ae32eae14864a3c9e6" alt=""
data:image/s3,"s3://crabby-images/23a56/23a56ce3971d7cc09e2e1c7adfdcaeedcba78d45" alt=""
data:image/s3,"s3://crabby-images/1484d/1484d51c2c9cd4018c96021655288ad714595b99" alt=""
3.1 版本 setnx
java
public String sale() {
String retMessage = "";
String key = "zzyRedisLock";
String uuidValue = IdUtil.simpleUUID() + Thread.currentThread().getId();
Boolean flag = stringRedisTemplate.opsForValue().setIfAbsent(key, uuidValue);
if (!flag) {
// 暂停20s,进行递归重试
try {TimeUnit.MICROSECONDS.sleep(20);} catch (InterruptedException e) {e.printStackTrace();}
sale();
} else {
// 抢锁成功的请求线程,进行正常的业务逻辑操作,扣减库存
try {
//1 查询库存信息
String result = stringRedisTemplate.opsForValue().get("inventory001");
//2 判断库存是否足够
Integer inventoryNumber = result == null? 0 : Integer.parseInt(result);
//3 扣除库存,每次减少一个
if (inventoryNumber > 0) {
inventoryNumber--;
stringRedisTemplate.opsForValue().set("inventory001", String.valueOf(inventoryNumber));
retMessage = "成功卖出一个商品,剩余库存" + inventoryNumber;
System.out.println(retMessage + ",端口号:" + port);
}else {
retMessage = "商品卖完了";
}
} finally {
stringRedisTemplate.delete(key);
}
}
return retMessage + ",端口号:" + port;
}
data:image/s3,"s3://crabby-images/0f4ed/0f4ed3e115d2a726d2897f80dd7945826721f910" alt=""
3.2 自旋代替递归重试 while替代if
java
public String sale() {
String retMessage = "";
String key = "zzyRedisLock";
String uuidValue = IdUtil.simpleUUID() + Thread.currentThread().getId();
// 不用递归了,高并发下使用自旋替代递归重试,使用while
while (!stringRedisTemplate.opsForValue().setIfAbsent(key, uuidValue)) {
try {
TimeUnit.MICROSECONDS.sleep(20);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
// 抢锁成功的请求线程,进行正常的业务逻辑操作,扣减库存
try {
//1 查询库存信息
String result = stringRedisTemplate.opsForValue().get("inventory001");
//2 判断库存是否足够
Integer inventoryNumber = result == null ? 0 : Integer.parseInt(result);
//3 扣除库存,每次减少一个
if (inventoryNumber > 0) {
inventoryNumber--;
stringRedisTemplate.opsForValue().set("inventory001", String.valueOf(inventoryNumber));
retMessage = "成功卖出一个商品,剩余库存" + inventoryNumber;
System.out.println(retMessage + ",端口号:" + port);
} else {
retMessage = "商品卖完了";
}
} finally {
stringRedisTemplate.delete(key);
}
return retMessage + ",端口号:" + port;
}
部署了微服务的Java程序挂了,代码层面根本没有走到 finally 这块,没办法保证解锁(无过期时间该key一直存在),这个key没有被删除,需要加入一个过期时间限定 key
4.1 setnx 添加过期时间
data:image/s3,"s3://crabby-images/82e67/82e678c1fe0c737613ee8616770a14b87b9333d4" alt=""
4.2 加锁和过期时间必须在同一行,保证原子性
data:image/s3,"s3://crabby-images/9e73e/9e73e723b8135ce802a375fada24741f83c1fdee" alt=""
5.0 防止误删key
锁超时误删除
data:image/s3,"s3://crabby-images/f0042/f0042ad24509b040124329d2b4bf44021b2b7811" alt=""
data:image/s3,"s3://crabby-images/7d07e/7d07e9e9f9d7fdfa2ce54a9d83c63733f3c99d32" alt=""
6.0 lua 脚本实现删除时判断和删除操作的原子性
data:image/s3,"s3://crabby-images/a63be/a63be8d758bd98cca6ebd0d8aa164e79cc0229b2" alt=""
介绍 lua 脚本
调用 lua 脚本,教程
- 实现 hello lua
注意 0 作为唤醒词
data:image/s3,"s3://crabby-images/3b8d8/3b8d8e4410657075e4d76e0023551e9b21d3de5f" alt=""
- 实现 set expire get
redis.call 用来调用命令,最后一个需要返回使用 return
data:image/s3,"s3://crabby-images/6c9a1/6c9a1f3b18c9ccde3afff63d211c40846adf7e06" alt=""
data:image/s3,"s3://crabby-images/8e0c4/8e0c4fe95701a172a36f7c0420480c303b0f6212" alt=""
- mset 重在掌握 lua脚本中 args传参的用法
2 -> key argv 数量
k1 k2 -> key1 key2 前两个 key,后面都是 argv
lua1 lua2 -> argv1 argv2
data:image/s3,"s3://crabby-images/4ba6c/4ba6c0de9b67dd6fce8cf1b9382a98424c62f8cf" alt=""
- 官网lua脚本 如何执行呢
lua
if redis.call("get",KEYS[1]) == ARGV[1] then
return redis.call("del",KEYS[1])
else
return 0
end
data:image/s3,"s3://crabby-images/586bb/586bb8049fdea4d55cbd36b21de0d7bcf38bd574" alt=""
补充说明 if elseif else
data:image/s3,"s3://crabby-images/f7392/f7392921122865640fc12303d55109d180c44dba" alt=""
data:image/s3,"s3://crabby-images/bdab0/bdab0e60db57ec7b67a38ab3c30354c5861f5cff" alt=""
java
// 改进点,修改为 Lua 脚本的 redis 分布式锁调用,必须保证原子性,参考官网脚本案例
String luaScript = "if redis.call('get',KEYS[1]) == ARGV[1] then " +
"return redis.call('del',KEYS[1]) " +
"else " +
"return 0 " +
"end";
stringRedisTemplate.execute(new DefaultRedisScript(luaScript, Boolean.class), Arrays.asList(key), uuidValue);
new DefaultRedisScript(luaScript, Boolean.class)
->Boolean.class
指定返回值,不指定会报错
data:image/s3,"s3://crabby-images/69d4c/69d4cdd021ce1f2fda1ead339796321174fe1bd7" alt=""
7.0 可重入 + 设计模式
juc 中可重入的解释
data:image/s3,"s3://crabby-images/108c6/108c63af1cb5004f78f69c53d6ea703fbb03c41d" alt=""
可重入锁的解释
data:image/s3,"s3://crabby-images/e0a1a/e0a1a6552ff80697e472aa38c44d631b8c983601" alt=""
juc 中可重入锁的案例演示(同步代码块,同步方法,Lock重点)
data:image/s3,"s3://crabby-images/e629e/e629ef47cf0ffbbdb649c03f81e387955fa1cdf2" alt=""
同步代码块
data:image/s3,"s3://crabby-images/e57d5/e57d50fe0b42589cbd0f86052174136757548bfb" alt=""
同步方法
data:image/s3,"s3://crabby-images/efe3f/efe3f29e553424358ae53ca7fbf3df48fb243bac" alt=""
注意使用 Lock 的时候,重入的时候加锁解锁次数要匹配,解锁次数不足则会导致锁仍然被占用,记得这里有个计数器
data:image/s3,"s3://crabby-images/c9a49/c9a494291297935b0d606eee3a3f90c9ebf0078b" alt=""
data:image/s3,"s3://crabby-images/4cf14/4cf140d00b906710727f61a8f597d78f07948a7d" alt=""
data:image/s3,"s3://crabby-images/b0faf/b0fafa75fcd35fefd99980313a702bfc2b121f49" alt=""
redis 中如何实现 aqs 中可重入规范
data:image/s3,"s3://crabby-images/84a56/84a5642b02f4e0fa3f32dd10f6971890e5891e71" alt=""
需要一个计数器,需要 kkv 这种结构才能满足 -> hset
data:image/s3,"s3://crabby-images/e679d/e679d9b64a1972a435872b6afdee728b1041ccce" alt=""
实现思路分析
data:image/s3,"s3://crabby-images/737c3/737c3238bce27670e8188d3a02a3fe884a3aa168" alt=""
data:image/s3,"s3://crabby-images/dafe8/dafe8791a3c895ceab6a27dcf49aa2480cada4d5" alt=""
lua 脚本分析
加锁Lock
- 判断redis分布式锁key是否存在 EXISTS key
- 不存在,hset新建当前线程属于自己的锁 uuid:ThreadId
- 存在,说明已经有锁 HEXISTS key uuid:ThreadId
- 不是自己的
- 是自己的,自增一次
lua
-- 加锁的lua的脚本,对标我们的lock方法
-- 加锁 v1
if redis.call('exists', 'key') == 0 then
redis.call('hset', 'key', 'uuid:threadid', 1)
redis.call('expire', 'key', '50')
return 1
elseif redis.call('hexists', 'key', 'uuid:threadid') == 1 then
redis.call('hincrby', 'key', 'uuid:threadid', 1)
redis.call('expire', 'key', '50')
return 1
else
return 0
end
-- v2 合并相同的代码,用incrby替代hset 简化代码
if redis.call('exists', 'key') == 0 or redis.call('hexists', 'key', 'uuid:threadid') == 1 then
redis.call('hincrby', 'key', 'uuid:threadid', 1)
redis.call('expire', 'key', '50')
return 1
else
return 0
end
-- v3 脚本ok了, 替换参数
if redis.call('exists', KEYS[1]) == 0 or redis.call('hexists', KEYS[1], ARGV[1]) == 1 then
redis.call('hincrby', KEY[1], ARGV[1], 1)
redis.call('expire', KEY[1], ARGV[2])
return 1
else
return 0
end
-- 合并到一行
if redis.call('exists', KEYS[1]) == 0 or redis.call('hexists', KEYS[1], ARGV[1]) == 1 then redis.call('hincrby', KEY[1], ARGV[1], 1) redis.call('expire', KEY[1], ARGV[2]) return 1 else return 0 end
加锁 unLock
设计思路:解锁,还得是自己的锁
lua
-- v1 解锁
if redis.call('hexists', 'key', 'uuid:threadid') == 0 then
return nil
elseif redis.call('hincrby', 'key', 'uuid:threadid', -1) == 0 then
return redis.call('del', 'key')
else
return 0
end
-- v2 替换下参数
if redis.call('hexists', KEYS[1], ARGV[1]) == 0 then
return nil
elseif redis.call('hincrby', KEYS[1], ARGV[1], -1) == 0 then
return redis.call('del', KEYS[1])
else
return 0
end
给出测试代码
加锁四次,解锁四次
shell
EVAL "if redis.call('exists', KEYS[1]) == 0 or redis.call('hexists', KEYS[1], ARGV[1]) == 1 then redis.call('hincrby', KEYS[1], ARGV[1], 1) redis.call('expire', KEYS[1], ARGV[2]) return 1 else return 0 end" 1 zzyyRedisLock 001:1 50
EVAL "if redis.call('exists', KEYS[1]) == 0 or redis.call('hexists', KEYS[1], ARGV[1]) == 1 then redis.call('hincrby', KEYS[1], ARGV[1], 1) redis.call('expire', KEYS[1], ARGV[2]) return 1 else return 0 end" 1 zzyyRedisLock 001:1 50
EVAL "if redis.call('exists', KEYS[1]) == 0 or redis.call('hexists', KEYS[1], ARGV[1]) == 1 then redis.call('hincrby', KEYS[1], ARGV[1], 1) redis.call('expire', KEYS[1], ARGV[2]) return 1 else return 0 end" 1 zzyyRedisLock 001:1 50
EVAL "if redis.call('exists', KEYS[1]) == 0 or redis.call('hexists', KEYS[1], ARGV[1]) == 1 then redis.call('hincrby', KEYS[1], ARGV[1], 1) redis.call('expire', KEYS[1], ARGV[2]) return 1 else return 0 end" 1 zzyyRedisLock 001:1 50
ttl zzyyRedisLock
HGET zzyyRedisLock 001:1
EVAL "if redis.call('hexists', KEYS[1], ARGV[1]) == 0 then return nil elseif redis.call('hincrby', KEYS[1], ARGV[1], -1) == 0 then return redis.call('del', KEYS[1]) else return 0 end" 1 zzyyRedisLock 001:1
EVAL "if redis.call('hexists', KEYS[1], ARGV[1]) == 0 then return nil elseif redis.call('hincrby', KEYS[1], ARGV[1], -1) == 0 then return redis.call('del', KEYS[1]) else return 0 end" 1 zzyyRedisLock 001:1
EVAL "if redis.call('hexists', KEYS[1], ARGV[1]) == 0 then return nil elseif redis.call('hincrby', KEYS[1], ARGV[1], -1) == 0 then return redis.call('del', KEYS[1]) else return 0 end" 1 zzyyRedisLock 001:1
EVAL "if redis.call('hexists', KEYS[1], ARGV[1]) == 0 then return nil elseif redis.call('hincrby', KEYS[1], ARGV[1], -1) == 0 then return redis.call('del', KEYS[1]) else return 0 end" 1 zzyyRedisLock 001:1
lua脚本整合进程序->实现 Lock 接口
data:image/s3,"s3://crabby-images/1eecf/1eecf55b942c8e2b2043bfc8f00d457453a37a10" alt=""
我们创建一个 Lock 实现类,重写lock(重载到 tryLock 中最终实现),unlock方法
java
@Override
public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
if (time == -1L) {
String luaScript = "if redis.call('exists', KEYS[1]) == 0 or redis.call('hexists', KEYS[1], ARGV[1]) == 1 then " +
" redis.call('hincrby', KEYS[1], ARGV[1], 1) " +
" redis.call('expire', KEYS[1], ARGV[2]) " +
" return 1 " +
"else" +
" return 0 " +
"end";
while (!(Boolean) stringRedisTemplate.execute(new DefaultRedisScript(luaScript, Boolean.class),
Arrays.asList(lockName), uuidValue, String.valueOf(expireTime))){
try {
Thread.sleep(60);
}catch (InterruptedException e) {
e.printStackTrace();
}
}
return true;
}
return false;
}
@Override
public void unlock() {
String luaScript = "if redis.call('hexists', KEYS[1], ARGV[1]) == 0 then " +
" return nil " +
"elseif redis.call('hincrby', KEYS[1], ARGV[1], -1) == 0 then " +
" return redis.call('del', KEYS[1]) " +
"else " +
" return 0 " +
"end";
// nil = false 1 = true 0 = false
// 这里返回值不方便使用 Boolean
Long flag = (Long) stringRedisTemplate.execute(new DefaultRedisScript(luaScript, Long.class),
Arrays.asList(lockName), uuidValue);
if (null == flag) {
throw new RuntimeException("锁不存在");
}
}
7.1 工厂模式优化,方便整合其他锁实现
data:image/s3,"s3://crabby-images/52612/5261202aa2b40ea333c5db0a29dbf382127b27eb" alt=""
data:image/s3,"s3://crabby-images/69371/69371f785a5a47fbb4db5b03f3e8980cde6dc498" alt=""
锁的实现写死了,以后如果引入别的类型的锁,zookeeper 等
java
@Component
public class DistributedLockFactory {
@Autowired
private StringRedisTemplate stringRedisTemplate;
private String lockName;
public Lock getDistributedLock(String lockType) {
if (lockType == null) {
return null;
}
if (lockType.equalsIgnoreCase("REDIS")) {
this.lockName = "zzyRedisLock";
return new RedisDistributedLock(stringRedisTemplate, lockName);
}else if (lockType.equalsIgnoreCase("ZOOKEEPER")) {
this.lockName = "zzyZookeeperLock";
// TODO zookeeper lock
return null;
} else if (lockType.equalsIgnoreCase("MYSQL")) {
// TODO mysql lock
}
return null;
}
}
7.2 可重入测试重点->解决uuid不一致,最终实现可重入
data:image/s3,"s3://crabby-images/19c58/19c58904af9d4f33467691b8acd6656630b2dea7" alt=""
data:image/s3,"s3://crabby-images/4b85c/4b85cdb0a41b67f7975b81305b48c8cfe854997b" alt=""
解决重入后 uuid 不一致问题
uuid 直接从工厂获取
data:image/s3,"s3://crabby-images/d5b0e/d5b0ee9e5a99836b25a0b9f4cb4a1750641d9ade" alt=""
这块说的比较碎,结合起来理解下
8.0 自动实现锁续期
CAP
data:image/s3,"s3://crabby-images/2a685/2a685f75161250dfcd76837a1a7b01dd817dbeb8" alt=""
Redis 可能存在异步复制导致的锁丢失,主节点还没来得及把刚刚set进来的这条数据同步到从节点,master就挂了,从机上位,但从机上无该数据
data:image/s3,"s3://crabby-images/9a4b7/9a4b78ba5aea534ddd2e9528cb16f62a347bd3ca" alt=""
zookeeper 只有整个同步全部成功,才会注册成功,速度会稍慢。主节点挂了,只有选举出新老大后,才可用
data:image/s3,"s3://crabby-images/63614/63614dcecc2e266a98896d615620e2952fdcd0fb" alt=""
euraka
加钟的 lua
在拿到锁的同时添加一个后台程序,为ttl 的 1/3
续期后再调用自己添加 监视程序,保证能够一直续期
lua
if redis.call('hexists', KEYS[1], ARGV[1]) == 1 then
return redis.call('expire', KEYS[1], ARGV[1])
else
return 0
end
代码中实现->加一个延时定时器任务
递归调用,3/1 时间间隔,如果锁仍然持有的话
比如 expireTime = 30,第20秒触发,继续续到 30
java
// tryLock 返回之前
// 新建一个后台扫描程序,来监视key目前的ttl,是否到我们规定的 1/2 1/3 来实现续期
resetExpire();
public void resetExpire() {
String script = "if redis.call('hexists', KEYS[1], ARGV[1]) == 1 then " +
"return redis.call('expire', KEYS[1], ARGV[2]) " +
"else " +
"return 0 " +
"end";
new Timer().schedule(new TimerTask() {
@Override
public void run() {
if ((boolean) stringRedisTemplate.execute(new DefaultRedisScript(script, Boolean.class),
Arrays.asList(lockName), uuidValue, String.valueOf(expireTime))) {
resetExpire();
}
}
}, (this.expireTime * 1000) / 3);
}
小总结&如果让你自研一把分布式锁,你要怎么做
- 按照 juc Lock 接口规范编写
- lock 加锁关键逻辑
- 加锁
- 给key设置一个值,为避免死锁,并给定一个过期时间
- 自旋
- 续期
- 加锁
data:image/s3,"s3://crabby-images/9b751/9b7513fbc607371632d4718247efa25a0dff3d78" alt=""
data:image/s3,"s3://crabby-images/4d38a/4d38a5cba735fca9b3e392a35236c977a74dd0ac" alt=""
- unlock 解锁关键逻辑
- 将 key 键删除,但也不能乱删,不能删除其他客户端的锁
data:image/s3,"s3://crabby-images/491b0/491b0fe903d044acec9d7565f834b611ae1f4d39" alt=""
redlock与 redisson
来由->我们手写的分布式锁有什么缺点
redis 单机故障
data:image/s3,"s3://crabby-images/b0bb2/b0bb29a6fb71681b8ad52cc9c68bf15954c86100" alt=""
data:image/s3,"s3://crabby-images/40acd/40acd884f730d43d50bd68e23aa29efc9c3747d7" alt=""
RedLock算法设计理念
data:image/s3,"s3://crabby-images/e9a67/e9a67f76387180ae39c58cb93604fbedb2467fa7" alt=""
data:image/s3,"s3://crabby-images/d436f/d436f36897d525e412660fa0efced068af62a1dd" alt=""
data:image/s3,"s3://crabby-images/1ebe2/1ebe28699fce8ca79789e314a053dcebd0772f99" alt=""
data:image/s3,"s3://crabby-images/14997/14997c7483a3920301f67bc733cfca01afb77d9a" alt=""
容错公式
data:image/s3,"s3://crabby-images/04f2d/04f2d255afe4251d870fd724d143525829439dd2" alt=""
v9.0
data:image/s3,"s3://crabby-images/a74a9/a74a9cad5b2bc7af67cb958f2497d02e95ad4ca4" alt=""
Getting Started - Redisson Reference Guide
xml
<dependency>
<groupId>org.redisson</groupId>
<artifactId>redisson</artifactId>
<version>3.20.1</version>
</dependency>
java
@Bean
public Redisson redisson() {
Config config = new Config();
config.useSingleServer().setAddress("redis://127.0.0.1:6379").setDatabase(0);
return (Redisson) Redisson.create(config);
}
RLock lock = redisson.getLock(key);
看门狗,续期
v9.1
data:image/s3,"s3://crabby-images/bfef1/bfef15eb254487652a553bc49d721ed3a5407407" alt=""
data:image/s3,"s3://crabby-images/d385b/d385b18197444119c6f96fdfa771d5feccfdc851" alt=""
Redisson 源码分析
data:image/s3,"s3://crabby-images/55d07/55d0701aa57d7cfd1f114e3e348f90b9ef107020" alt=""
data:image/s3,"s3://crabby-images/f9eff/f9effa64073a1bd4673497bfc5d45beb31a4bb50" alt=""
data:image/s3,"s3://crabby-images/3d7bf/3d7bfdd40f16c5989dd0a1ed9642b8c7aded01c2" alt=""
data:image/s3,"s3://crabby-images/a95dc/a95dc64029a4aef19ffebb8372fefacc68cd7b41" alt=""
data:image/s3,"s3://crabby-images/4cc0b/4cc0ba17fb2a126ed6aebc2e713c6397a0c0a0ec" alt=""
data:image/s3,"s3://crabby-images/6e55f/6e55f01364b12e39fdba5a268c1bcff5c49b6d35" alt=""
data:image/s3,"s3://crabby-images/0fd1a/0fd1acdadda46d805cf72df097e09ee8f65869fe" alt=""
data:image/s3,"s3://crabby-images/fd07c/fd07c24c40c035f18cb8502e2186ad7db09e2cee" alt=""
data:image/s3,"s3://crabby-images/43174/43174488bbd9bef5d5b19a79c8b6cb68caad6ed7" alt=""
data:image/s3,"s3://crabby-images/26d8c/26d8c7d22dbdd6a8298be640a6d217feb2d8d75c" alt=""
多机案例->MultiLock 创建联锁
data:image/s3,"s3://crabby-images/489cd/489cd0b45ae2d662d1533254a4c85a3c4c3f3641" alt=""
data:image/s3,"s3://crabby-images/66aa5/66aa5edbd387021bea7dd96b8b4a9a6f4fadd44a" alt=""
但 RedLock 被标记弃用
data:image/s3,"s3://crabby-images/96121/96121c63fcdf87a618feda8beee50fbecf583656" alt=""
将多个 RLock 对象关联为一个联锁对象,只有所有节点加锁成功,联锁才成功
实操
data:image/s3,"s3://crabby-images/5c76d/5c76dcc24c65e32731d93addcc42a0c01ce631b8" alt=""
data:image/s3,"s3://crabby-images/0b8d5/0b8d5263a0c0399817cbef71fbb5cdcd57add9c4" alt=""
data:image/s3,"s3://crabby-images/6c12d/6c12de22470febd55bd137be686d9af7ecb6a603" alt=""
data:image/s3,"s3://crabby-images/a70de/a70de12f5cada5ab399f51e064e25f1ddb5deb7b" alt=""
data:image/s3,"s3://crabby-images/b5996/b5996ccf685bbc00741f530c39a3c87ab74a4822" alt=""
data:image/s3,"s3://crabby-images/1dc5b/1dc5b78b7ff9b635432c5fd9bca764cf8774a2d4" alt=""
data:image/s3,"s3://crabby-images/a0b35/a0b350267dc90b5194f1b2f40b96cb0aa2135ca6" alt=""
我们创建三个 RedisClient,然后创建一个联锁即可
data:image/s3,"s3://crabby-images/84f30/84f309ac3f1bc924333b097abd2e6fe00aa654ba" alt=""