简介
本文注解组件的作用:方法上加个注解就能使用Redisson分布式锁。
为什么使用分布式锁?
为了防止重复执行。以退款为例,每笔订单只能退一次款。代码逻辑:先判断数据库是否已经退款,如果还没退款则允许执行,执行结束后将退款记录到数据库内。
假设此时用户点击了两次,项目使用提交读或者重复读,在第一个请求还没执行结束时,第二个请求进来了,由于第一个请求的数据库事务还没提交,所以第二个请求还查不到第一个请求的退款记录,所以第二个请求也执行了。最终退了两次款!
解决方案就是:使用分布式锁,通过Redis来记录。如果Redis中还没有,表示还没有请求,将数据写到Redis,然后执行代码,执行结束后删除Redis数据。
本文注解支持的功能:
- 可以指定要加锁的字段
- 可以指定获取锁的超时时间
- 可以指定获取锁失败时的提示信息
- 可以指定获取锁失败时抛出的异常
注意:本组件是对Redisson的封装,Redisson的分布式锁是最完美的,对它进行封装即可如虎添翼!
虽然网上有开源的了,比如:lock4j,但它有bug,执行完后不释放锁,根本没法用!
Java初级和高级的做法
Java初级开发的做法
Java初级开发者是这样使用Redisson分布式锁的:
- 手动加锁
- 将原逻辑放到获取锁成功后的代码块内
- 一般锁是在@Transactional之前执行,所以要将原逻辑封装到一个方法中,然后在锁获取成功后调用它
- 逻辑执行完后要释放锁
Java高级开发的做法
Java高级开发一眼就能看出问题了:这都是重复性的东西,提取出公共组件,让AOP去处理,原逻辑上边只加个注解即可。
效果展示
Controller
本处为了简单,直接放到了Controller上,实际应该放到Service上。
package com.knife.example.business.product.controller;
import com.knife.example.business.product.bo.ProductSaveBO;
import com.knife.example.business.product.bo.UserBO;
import com.knife.example.common.annotation.DistributionLock;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import org.springframework.web.bind.annotation.*;
@Api(tags = "商品")
@RestController
@RequestMapping("product")
public class ProductController {
@ApiOperation("新增")
@DistributionLock(keys = {"#productSaveBO.id", "#productSaveBO.name"})
@PostMapping("add")
public String add(@RequestBody ProductSaveBO productSaveBO) {
try {
Thread.sleep(10000);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
return "success";
}
@ApiOperation("编辑")
// 支持多个入参
@DistributionLock(keys = {"#productSaveBO.id", "#productSaveBO.name"})
@PostMapping("edit")
public String edit(@RequestBody ProductSaveBO productSaveBO,
UserBO userBO) {
try {
Thread.sleep(10000);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
return "success";
}
}
测试
访问:http://localhost:8080/doc.html(打开两个页面)
两个页面都输入入参,点击完一个页面的“发送”后,快速点击另一个页面的“发送”,模拟并发场景。

结果
第1个页面:(获取到锁,正在执行)

第2个页面:(获取锁失败,报错)

后端结果:

等第一个页面执行结束:(执行成功,释放锁)

Redis里的数据
组件自动加锁时,往Redis里写的数据是这样的:

锁释放后,Redis这个key就没了:

代码
此内容仅限VIP查看,请先登录

请先 !