SpringBoot自定义注解AOPredis实现防接口重复提交,概念到实战
一、前言
在面试中,经常会有一道经典面试题,那就是: 怎么防止接口重复提交?
小编也是背过的,好几种方式,但是一直没有实战过,做多了管理系统,发现这个事情真的没有过多的重视。
最近在测试过程中,发现了多次提交会保存两条数据,进而导致程序出现问题!
问题已经出现我们就解决一下吧!!
本次解决是对于高并发不高的情况,适用于一般的管理系统,给出的解决方案!!高并发的还是建议加分布式锁!!
下面我们来聊聊幂等性是什么? 二、什么是幂等性
接口幂等性就是用户对于 同一操作 发起的一次请求或者多次请求 的结果是一致的 ,不会因
为多次点击而产生了副作用;
比如说经典的支付场景:用户购买了商品支付扣款成功,但是返回结果的时候网络异常,此时钱已经扣了,用户再次点击按钮,此时会进行第二次扣款,返回结果成功,用户查询余额返发现多扣钱了,流水记录也变成了条,这就没有保证接口的幂等性;
可谓:商家美滋滋,买家骂咧咧!!
防接口重复提交,这是必须要做的一件事情!! 三、REST风格与幂等性
以常用的四种来分析哈!
REST
是否支持幂等
SQL例子
GET
是
SELECT * FROM table WHER id = 1
PUT
是
UPDATE table SET age=18 WHERE id = 1
DELETE
是
DELETE FROM table WHERE id = 1
POST
否
INSERT INTO table (id,age) VALUES(1,21)
所以我们要解决的就是 POST 请求!四、解决思路
大概主流的解决方案: token机制(前端带着在请求头上带着标识,后端验证) 加锁机制 数据库悲观锁(锁表) 数据库乐观锁(version号进行控制) 业务层分布式锁(加分布式锁redisson) 全局唯一索引机制 redis的set机制 前端按钮加限制
小编的解决方案就是redis的set机制!
同一个用户,任何POST保存相关的接口,1s内只能提交一次。
完全使用后端来进行控制,前端可以加限制,不过体验不好!
后端通过自定义注解,在需要防幂等接口上添加注解,利用AOP切片,减少和业务的耦合!
在切片中获取用户的 token、user_id、url 构成redis的唯一key!
第一次请求会先判断key是否存在,如果不存在,则往redis添加一个主键key,设置过期时间;
如果有异常会主动删除key,万一没有删除失败,等待1s,redis也会自动删除,时间误差是可以接受的!
第二个请求过来,先判断key是否存在,如果存在,则是重复提交,返回保存信息!! 五、实战
SpringBoot版本为 2.7.4 1. 导入依赖 org.springframework.boot spring-boot-starter-data-redis org.projectlombok lombok 1.18.2 org.springframework.boot spring-boot-starter-aop org.springframework.boot spring-boot-starter-web com.alibaba druid-spring-boot-starter 1.1.16 org.springframework.boot spring-boot-starter-jdbc mysql mysql-connector-java com.baomidou mybatis-plus-boot-starter 3.5.1 org.springframework.boot spring-boot-starter-test test 2. 编写ymlserver: port: 8087 spring: redis: host: localhost port: 6379 password: 123456 datasource: #使用阿里的Druid type: com.alibaba.druid.pool.DruidDataSource driver-class-name: com.mysql.cj.jdbc.Driver url: jdbc:mysql://127.0.0.1:3306/test?serverTimezone=UTC username: root password: 3. redis序列化/** * @author wangzhenjun * @date 2022/11/17 15:20 */ @Configuration public class RedisConfig { @Bean @SuppressWarnings(value = { "unchecked", "rawtypes" }) public RedisTemplate