• 114 接口中幂等性的保证


    前言

    同样是 面试问题 

    如何确保接口的 幂等性 

    幂等是一个 较为抽象的概念, 多次重复访问, 不会导致业务逻辑的异常 

    这里从增删改查, 几个方面列一下 

    一般来说, 我们核心需要关注的就是 新增 和 更新

    对于 增加元素, 首先针对唯一约束进行校验, 然后再处理新增的相关业务, 严格一点需要 加锁, 分布式并发控制 

    对于 删除元素, 就是检查元素存不存在, 存在 则删除, 不存在 返回相关状态吗, 或者直接成功都 OK

    元素的新增

    基于持久化的数据库的机制

    比如 mysql 这边目标表, 增加唯一索引, 或者 主键

    比如, 我们这里 限定在 用户表 中 用户名 不能重复, 这个只有特定的业务场景中可以这么处理 

    1. CREATE TABLE `auth_user` (
    2. `id` int(11) NOT NULL,
    3. `name` varchar(256) DEFAULT NULL,
    4. `age` int(11) DEFAULT NULL,
    5. PRIMARY KEY (`id`),
    6. UNIQUE KEY `name` (`name`) USING BTREE COMMENT 'name'
    7. ) ENGINE=InnoDB DEFAULT CHARSET=utf8

    然后 服务器这边 就不用做 过多的控制, 核心业务部分直接 ”insert into” 都可以 

    由 mysql 这边本身的机制 来确保 用户名 的不能重复, 防止 用户多次提交 造成的业务问题

    分布式并发过滤控制 + 数据库的悲观锁

    我们这里展现一下 完整的处理流程, 主要是包含了 外层的并发过滤控制, 数据库校验控制, 数据库加锁+插入 控制

    这里分布式并发控制这里模拟实现, 是 userRunningStore 部分

    1. 并发过滤控制这边处理如下, 基于 spring 的 注解 + aop

    1. @Target(ElementType.METHOD)
    2. @Retention(RetentionPolicy.RUNTIME)
    3. public @interface ConcurrentLatch {
    4. }

    并发过滤控制的处理如下

    1. @Component
    2. @Aspect
    3. public class ConcurrentLatchAop {
    4. @Pointcut("@annotation(ConcurrentLatch)")
    5. public void concurrentLatchAop() {
    6. }
    7. Set<String> userRunningStore = new LinkedHashSet<>();
    8. @Around("concurrentLatchAop()")
    9. public Object doProcess(ProceedingJoinPoint point) throws Throwable {
    10. Object[] args = point.getArgs();
    11. AuthUser user = (AuthUser) args[0];
    12. String name = user.getName();
    13. // lock
    14. if (userRunningStore.contains(name)) {
    15. throw new RuntimeException(String.format("有其他用户在新增用户 %s, 请刷新后重试", name));
    16. }
    17. userRunningStore.add(name);
    18. // unlock
    19. Object result = point.proceed();
    20. return result;
    21. }
    22. }

    数据库校验控制, 数据库加锁+插入控制 如下 

    1. @PutMapping("/user")
    2. @ConcurrentLatch
    3. public AuthUser add(AuthUser user) {
    4. // physic verify
    5. if (user.getAge() > 0 && user.getAge() < 111) {
    6. throw new RuntimeException("用户的 age 必须在合法的区间");
    7. }
    8. // logistic verify
    9. Map<String, Object> existsUser = JdbcTemplateUtils.queryOne(jdbcTemplate, String.format(" select * from auth_user where name = '%s'; ", user.getName()));
    10. if (existsUser != null) {
    11. throw new RuntimeException("该用户已经存在, 用户名称不能重复");
    12. }
    13. // do other biz
    14. // lock then insert
    15. existsUser = JdbcTemplateUtils.queryOne(jdbcTemplate, String.format(" select * from auth_user where name = '%s' for update; ", user.getName()));
    16. if (existsUser != null) {
    17. throw new RuntimeException("该用户已经存在, 用户名称不能重复");
    18. }
    19. jdbcTemplate.execute(String.format("INSERT INTO `auth_user`(`name`, `age`) VALUES ('%s', %s);", user.getName(), user.getAge()));
    20. return user;
    21. }

    token分布式并发控制 + 数据库的悲观锁

    这个就主要是 整体的交互机制调整, 增加了一层 token 的获取 和 验证

    token 的分派这边如下, 做限流, 生成 token 的相关处理 

    1. public static Map<String, AtomicInteger> interf2Counter = new LinkedHashMap<>();
    2. public static Set<String> tokenStore = new LinkedHashSet<>();
    3. // pre install all interfs
    4. static {
    5. interf2Counter.put("IdempotentController.add", new AtomicInteger());
    6. }
    7. @GetMapping("/requestToken")
    8. public String requestToken(String interf) {
    9. AtomicInteger counter = interf2Counter.get(interf);
    10. int incred = counter.getAndIncrement();
    11. // rate limit
    12. if (incred > 20) {
    13. counter.getAndDecrement();
    14. throw new RuntimeException(" 服务器繁忙, 请稍后重试 ");
    15. }
    16. String token = UUID.randomUUID().toString();
    17. String compositeToken = interf + token;
    18. tokenStore.add(compositeToken);
    19. return token;
    20. }

    并发控制这边处理如下 

    1. /**
    2. * ConcurrentLatchAop
    3. *
    4. * @author Jerry.X.He
    5. * @version 1.0
    6. * @date 2023/9/21 10:17
    7. */
    8. @Component
    9. @Aspect
    10. public class ConcurrentLatchAop {
    11. @Pointcut("@annotation(ConcurrentLatch)")
    12. public void concurrentLatchAop() {
    13. }
    14. @Around("concurrentLatchAop()")
    15. public Object doProcess(ProceedingJoinPoint point) throws Throwable {
    16. Object[] args = point.getArgs();
    17. String interf = "get interf from request";
    18. String token = "get token from request";
    19. // lock
    20. if (!IdempotentController.tokenStore.contains(token)) {
    21. throw new RuntimeException("服务器异常, 请刷新后重试");
    22. }
    23. IdempotentController.tokenStore.remove(token);
    24. // unlock
    25. Object result = point.proceed();
    26. AtomicInteger counter = IdempotentController.interf2Counter.get(interf);
    27. counter.getAndDecrement();
    28. return result;
    29. }
    30. }

    元素的更新

    以上 三种处理方式 在元素的更新中同样可以使用

    元素的更新 数据库的更新控制这边可以使用 基于数据库的乐观锁 

    数据库的乐观锁更新

    并发控制这边 和上面类似, 我们这里着重关注 数据库的更新这边 

    数据库的更新这边, 主要是增加一个版本号的字段, 然后 更新的时候 在原有的 id 条件之外, 再增加一个 version 控制的字段 

    根据 mysql 这边更新, 会增加行排他锁, 具体的处理如下 

    1. /**
    2. * IdempotentController
    3. *
    4. * @author Jerry.X.He
    5. * @version 1.0
    6. * @date 2023/9/21 9:58
    7. */
    8. @RestController
    9. @RequestMapping("/idempotent")
    10. public class IdempotentController {
    11. @Resource
    12. private JdbcTemplate jdbcTemplate;
    13. @PostMapping("/user")
    14. @ConcurrentLatch
    15. public AuthUser update(AuthUser user) {
    16. // physic verify
    17. if (user.getAge() > 0 && user.getAge() < 111) {
    18. throw new RuntimeException("用户的 age 必须在合法的区间");
    19. }
    20. // logistic verify
    21. Map<String, Object> existsUser = JdbcTemplateUtils.queryOne(jdbcTemplate, String.format(" select * from auth_user where name = '%s'; ", user.getName()));
    22. if (existsUser == null) {
    23. throw new RuntimeException("该用户不存在, 请确认输入");
    24. }
    25. // do other biz
    26. // lock then insert
    27. String id = String.valueOf(existsUser.get("id"));
    28. String version = String.valueOf(existsUser.get("version"));
    29. int updatedCount = jdbcTemplate.update(String.format("update auth_user set name = '%s', age = %s where id = %s and version = %s;", user.getName(), user.getAge(), id, version));
    30. if (updatedCount == 0) {
    31. throw new RuntimeException("该用户信息已经发生改变, 请刷新后重试");
    32. }
    33. return user;
    34. }
    35. }

  • 相关阅读:
    vue中倒计时(日,时,分,秒)的计算和当前时间计时读秒
    ffmpeg转码生成的m3u8格式详解
    jenkins 使用原生 git clone 命令,指定ssh密钥文件
    Linux命令之常用基础命令备查手册
    上下文管理器与with关键字为业务增加辅助功能
    Unix Network Programming Episode 80
    PHP和JAVA AES加解密问题
    决策树 #数据挖掘 #Python
    @AutoConfigurationPackage注解类
    图像处理ASIC设计方法 笔记27 红外非均匀校正的两点定标校正算法
  • 原文地址:https://blog.csdn.net/u011039332/article/details/133394605