logo头像

科学上网工具

To be a better man !

Java基于Spring AOP+Redis+注解实现适用多种场景的分布式锁

本文于1665天之前发表,文中内容可能已经过时。

分布式系统开发中常常用到分布式锁,比如防止多个用户同时预订同一个商品,传统的synchronized就无法实现了,而基于数据库的乐观锁实现又可能会对数据库产生较大的压力。而分布式锁相对较轻量,对性能影响也较小。目前主流的分布式锁都基于Redis实现。使用分布式锁的流程一般如下:

YGxDoj.png

如果需要使用分布式锁的地方有多个,那么就需要写多个类似的代码。而重复代码是开发中最常见到的 bad smell 。我们可以使用 AOP 把这段逻辑抽象出来,这样就避免了重复代码,也极大地减去了工作量。

目标

  • 对业务代码无侵入(或侵入性较小)
  • 使用方便
  • 对性能影响小
  • 易维护

方案

  1. 使用注解(假设注解为@Lock)声明要使用分布式锁的业务method、要锁定的对象(一般是业务主键)、失效时间等信息。
  2. 使用Spring AOP arround(环绕通知)增强被@Lock注解的方法,把前面提到的”使用分布式锁的流程“逻辑抽象到切面中。
  3. 使用Redis实现分布式锁。一般是基于string类型的set命令实现。

难点

  1. 如何根据请求的不同,锁定不同的对象
    可以使用 Spring EL 表达式指定锁定对象,加锁时根据业务方法参数值、参数名称解析表达式,得出要加锁的对象(Redis string的key)。
  2. 分布式锁该如何选择
    • 可以选择自己实现(加锁使用set命令即可,解锁需要使用lua脚本保证命令的原子性,先判断锁是否仍有效、是否由当前线程加锁,是的话才能通过del来解锁)。
    • 也可以选择使用redisson等第三方库。使用方式可以参考官方示例:Spring版/Spring Boot版

优点

使用时只需在业务方法上加一个注解就可以了,使用灵活、开发效率高、侵入小、适用性强。业务方法只需专注于业务代码,可读性强,易维护。

适用场景

  • 防止并发修改
  • 防止重复提交
  • 幂等校验
  • ……

实现

1. 引入包

Spring Boot方式

1
2
3
4
5
6
7
8
9
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
<dependency>
<groupId>org.redisson</groupId>
<artifactId>redisson-spring-boot-starter</artifactId>
<version>3.12.5</version>
</dependency>

Spring 方式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>5.1.3.RELEASE</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.2</version>
</dependency>
<dependency>
<groupId>org.redisson</groupId>
<artifactId>redisson</artifactId>
<version>3.12.5</version>
</dependency>

2. 定义注解

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
@Documented
@Inherited
public @interface Lock {

/** 锁分组名称,用于避免key重复 */
String group() default "redis_lock_order";

/** key 表达式 */
String value();

/** 获取锁失败时的提示信息 */
String lockFailedMsg() default "当前订单正在修改中,请稍候重试";

/**
* 等待时长(ms),默认为0,获取不到锁立即返回。
* @return 等待时长 ms
*/
long waitTime() default 0;

/**
* 最大持有锁时间,如果超过该时间仍未主动释放,将自动释放锁。
* @return 最大持有锁时间 ms
*/
long leaseTime() default 5000;
}

3. 定义注解数据模型

1
2
3
4
5
6
7
8
9
10
11
12
13
@Value
public class LockVO {
/** key前缀 */
private String group;
/** key */
private String key;
/** 加锁失败时的提示信息 */
private String lockFailedMsg;
/** 等待时长 */
private Long waitTime;
/** 持有锁时长 */
private Long leaseTime;
}

4. 定义切面逻辑

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
@Aspect
@Component
@Slf4j
public class LockAspect {
/** 拦截所有注解了@Lock的方法 */
@Pointcut(value = "@annotation(lock)")
public void pointcut(Lock lock) {}

@Around(value = "pointcut(lock)", argNames = "jp,lock")
public Object aroundRemote(ProceedingJoinPoint jp, Lock lock) throws Throwable {
// 解析注解数据
LockVO lockVo = parseLockAnnotationData(jp, lock);
// 生成锁key
String lockKey = lockVo.getGroup() + ":" + lockVo.getKey();

// 竞争分布式锁
String lockId = lock(lockVo, lockKey);
try {
return jp.proceed();
} finally {
// 释放锁
unlock(lockKey, lockId);
}
}

private String lock(LockVO lockVo, String lockKey) {
// 具体加锁逻辑可以自己选择,推荐使用redisson,也可以选择自己实现。
// 自己实现需要考虑满足可重入性、锁超时等问题。
// 目前没有完美的分布式锁实现,需要根据自己的项目的应用场景做出权衡和选择。

// 以下介绍自己实现逻辑的大致思路
// 1. 如果当前线程已经拿到锁,直接返回
// 可以通过ThreadLocal实现可重入式分布式锁,具体实现省略。

// 2. 否则,获取redis分布式锁,拿到新的lockId
Long waitTime = lockVo.getWaitTime();
Long leaseTime = lockVo.getLeaseTime();

if (waitTime > 0) {
// 基于set命令实现,指定key失效时间,如果未获取到锁,则等待若干ms后重试,
// 在waitTime过后仍未获取到锁则获取锁失败。
lockId = RedisStringUtil.tryLock(lockKey, waitTime, leaseTime, TimeUnit.MILLISECONDS);
if (lockId == null) {
log.error("WARN:延迟加锁失败,数据可能出现问题,锁key:{},锁数据:{}",
lockKey, lockVo);
}
} else {
// 基于set命令实现,指定key失效时间,如果未获取到锁。则加锁失败。
lockId = RedisStringUtil.tryLock(RedisNamespaceEnum.LOCK.getValue(), lockKey, leaseTime, TimeUnit.MILLISECONDS);
}
// 未拿到锁
if (lockId == null) {
throw new RuntimeException(lockVo.getLockFailedMsg());
}
// 放入ThreadLocal,用于实现可重入性
return lockId;
}

private void unlock(String lockKey, String lockId) {
// 释放分布式锁
if(ReentrantUtil.canRemove(lockKey)){
ReentrantUtil.remove(lockKey);
// 如果lockId代表的锁依然存在,则可以解锁成功。
Boolean success = RedisStringUtil.unlock(lockKey, lockId);
if (!success) {
// 锁超时情况下会出现该问题,当出现该问题时,需要根据情况特殊处理。
log.error("释放锁失败,lockKey:{},lockId:{}", lockKey, lockId);
}
} else {
ReentrantUtil.release(lockKey);
}
}

/**
* 解析@Lock数据
*/
private LockVO parseLockAnnotationData(ProceedingJoinPoint jp, Lock lock) {
Method method = (MethodSignature)jp.getSignature();

String keyExpression = lock.value();
// 解析el表达式,获取锁key
String key = parseElExpression(jp.getArgs(), method, keyExpression, String.class);
return new LockVO(lock.group(), key, lock.lockFailedMsg(), lock.waitTime(), lock.leaseTime());
}

/**
* 解析EL表达式
* @param args 方法参数
* @param method 方法
* @param elExpression EL表达式
* @param resultType 结果类型
* @param <T> 结果类型
* @return 结果
*/
private static <T> T parseElExpression(Object[] args, Method method, String elExpression, Class<T> resultType) {
Parameter[] parameters = method.getParameters();
StandardEvaluationContext elContext = new StandardEvaluationContext();
if (parameters != null && parameters.length > 0) {
// 设置解析变量
for (int i = 0; i < parameters.length; i++) {
String paraName = parameters[i].getName();
Object paraValue = args[i];
elContext.setVariable(paraName, paraValue);
}
}
ExpressionParser parser = new SpelExpressionParser();
return parser.parseExpression(elExpression)
.getValue(elContext, resultType);
}

}

5. 打开编译开关

在项目的pom文件中的编译插件中添加参数:-parameters(JDK8+才支持),用于在编译时把方法参数名称保留到 class 文件中。这样我们就可以通过Spring EL表达式动态指定要加锁的key。

1
2
3
4
5
6
7
8
9
10
11
12
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.3</version>
<configuration>
<source>1.8</source>
<target>1.8</target>
<compilerArgs>
<arg>-parameters</arg>
</compilerArgs>
</configuration>
</plugin>

JDK8以下版本可以使用别的方式获取方法参数名称,也可以将表达式写为类似于
[0].getOrderId()(获取第一个入参的orderId属性的值)
的格式动态指定key(因为方法入参可以看做是一个Object[])。

6. 使用

  • 防止并发修改,可以把Lock注解在例如修改订单的接口方法上,waitTime设置为0,key为订单id,这样就可以防止多个人并发修改订单。

    1
    2
    3
    4
    5
    @Lock(group = "order_lock", value = "#dto.getOrderId()", 
    lockFailedMsg = "当前订单正在修改中,请稍候重试", leaseTime = 5000)
    public void modifyOrder(ModifyOrderDTO dto) {
    // ......
    }
  • 防止重复提交。例如防止重复下单,可以将waitTime设置为0,key为会员id,在订单未保存成功前,用户多次提交订单都会直接返回提示信息。

    1
    2
    3
    4
    5
    6
    @Lock(group = "forbid_repeat_submission_4_order", value = "#orderVO.getMemberId()",
    lockFailedMsg = "订单已提交,请稍候", leaseTime = 10000)
    @Transactional(rollbackFor = Exception.class)
    public OrderVO saveOrder(OrderVO orderVO) {
    // ......
    }

总结

基于注解的分布式锁可以帮助我们减少大量模板代码,使用方便,出现问题也很容易修复。对于具体的加锁逻辑可以选择自己实现,也可以选择使用redisson等第三方库。

微信打赏

赞赏是不耍流氓的鼓励