经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 数据库/运维 » MyBatis » 查看文章
MyBatisPlus解决逻辑删除与唯一索引的兼容问题
来源:cnblogs  作者:雨点的名字  时间:2023/4/14 9:19:28  对本文有异议

需求背景

比如有张用户表,在插入或者更新数据的时候,我们需要 用户名称(username),不能重复。

我们首先考虑的是给该字段创建唯一索引

  1. create unique index uni_username on user(username)

似乎这样就可以了,然而事情并没有那么简单。

因为我们表中的数据在删除的时候不会真的的删除,而是采用逻辑删除,会有一个 deleted 字段使用0,1标识未删除与已删除。

当然我们可以考虑将 username + deleted 组合成一个联合唯一索引。

  1. create unique index uni_username_deleted on user(username,deleted)

这样就ok了吗?

其实会有一个新的问题,就是如果同一个用户名如果被删除一次。

再去删除会发现系统报错了,因为该条数据已经存在了,不能在删除了。

是不是很多时候因为逻辑删除与唯一索引的冲突,你就不创建唯一索引,想着自己写的代码自己有信心不会出现脏数据的。

这么想你就太天真啦,数据库是我们最后一道防线,这道防线都不要了嘛?

阿里巴巴手册有关索引规范,第一条就是

  1. 【强制】业务上具有唯一特性的字段,即使是组合字段,也必须建成唯一索引。

手册还有这么一句话:

  1. 即使在应用层做了非常完善的校验和控制,只要没有唯一索引,根据墨菲定律,必然有脏数据产生。

所以唯一索引非常有必要!!!

那该怎么做能让逻辑删除与唯一索引兼容?

现在大家比较通用的办法就是

  1. 我们依旧可以将 username + deleted 组合成一个联合唯一索引,但是删除的时候deleted不再是固定的1,而是当前的主键ID,也就是deleted不等于0都是删除状态,如果删除了那deleted值=id

既然确立了解决方案,那就该思考怎么做?


二、MyBatisPlus逻辑删除

MyBatisPlus是支持逻辑删除的,如果确定在哪个字段是逻辑删除字段,那就在该字段上添加一个注解

  1. /**
  2. * 1、删除 0、未删除
  3. */
  4. @TableLogic(value = "0", delval = "1")
  5. private Integer deleted;

这个一来操作数据是会自动变成如下:

  • 查询时: 查询条件会自动加上 'AND deleted = 0'
  • 删除时: 自定添加 'UPDATE SET deleted = 1 … WHERE … AND deleted = 0'

如果你想删除的时候不再是固定1而是id值,那么就可以这样改

  1. @TableLogic(value = "0", delval = "id")
  2. private Integer deleted;

如果想改成全局的那么在配置文件中添加

  1. mybatis-plus:
  2. global-config:
  3. db-config:
  4. logic-delete-value: 1 # 逻辑已删除值(默认为 1)
  5. logic-not-delete-value: 0 # 逻辑未删除值(默认为 0)

三、测试

1、用户表

  1. CREATE TABLE `user` (
  2. `id` int unsigned AUTO_INCREMENT COMMENT '主键',
  3. `username` varchar(128) COMMENT '用户名',
  4. `phone` varchar(32) COMMENT '手机号',
  5. `sex` char(1) COMMENT '性别',
  6. `create_time` datetime COMMENT '创建时间',
  7. `update_time` datetime COMMENT '更新时间',
  8. `deleted` tinyint DEFAULT '0' COMMENT '1、删除 0、未删除',
  9. PRIMARY KEY (`id`)
  10. ) ENGINE=InnoDB AUTO_INCREMENT=1

2、创建对应实体

  1. @Data
  2. @Accessors(chain = true)
  3. @TableName("user")
  4. public class UserDO implements Serializable {
  5. private static final long serialVersionUID = 1L;
  6. @TableId(type = IdType.AUTO)
  7. private Integer id;
  8. /**
  9. * 用户名
  10. */
  11. private String username;
  12. /**
  13. * 手机号
  14. */
  15. private String phone;
  16. /**
  17. * 性别
  18. */
  19. private String sex;
  20. /**
  21. * 创建时间
  22. */
  23. private LocalDateTime createTime;
  24. /**
  25. * 更新时间
  26. */
  27. private LocalDateTime updateTime;
  28. /**
  29. * 1、删除 0、未删除
  30. */
  31. private Integer deleted;
  32. }

3、物理删除测试

注意: 目前 deleted 字段是没有添加 @TableLogic注解,同是在全局也没有定义逻辑删除

我们来看下删除示例

  1. @Test
  2. public void deleteById() {
  3. //方式一:根据id删除
  4. mapper.deleteById(10);
  5. //方式二:根据指定字段删除
  6. LambdaQueryWrapper<UserDO> wrapper = new LambdaQueryWrapper<>();
  7. wrapper.eq(UserDO::getSex, "男");
  8. mapper.delete(wrapper);
  9. //方式三:手动逻辑删除
  10. UserDO userDO = new UserDO();
  11. userDO.setId(10);
  12. userDO.setDeleted(1);
  13. mapper.updateById(userDO);
  14. }

执行结果

  1. --方式1
  2. DELETE FROM user WHERE id=10
  3. --方式2
  4. DELETE FROM user WHERE (sex = '男')
  5. --方式3
  6. UPDATE user SET deleted=1 WHERE id=10

我们通过结果可以看出,如果不添加逻辑删除标识 那删除就是物理删除。

4、逻辑删除测试

我们在deleted属性字段 添加 逻辑删除标识

  1. @TableLogic(value = "0", delval = "id")
  2. private Integer deleted;

我们再来执行上面三个删除,看下执行结果

  1. --方式1
  2. UPDATE user SET deleted=id WHERE id=10 AND deleted=0
  3. --方式2
  4. UPDATE user SET deleted=id WHERE deleted=0 AND (sex = '男')
  5. --方式3
  6. 报错了

从执行结果来看,方式一和方式二都从之前的物理删除变成了逻辑删除。

但为什么方式三会报错呢?我们来看下报错的结果

发现问题了,最终执行的SQL竟然是:

  1. UPDATE user WHERE id=? AND deleted=0

为什么是这样,正常不应该是

  1. UPDATE user SET deleted=1 WHERE id=? AND deleted=0

这个就需要去看Mybatisplus到底做了什么操作,改变了我们的SQL

真相大白了

  1. MybatisplusupdateById更新时,如果已经加了逻辑删除标记,那做SQL拼接的时候,会自动过滤掉逻辑删除的Set拼接

所以在实际开发中就非常注意,如果你的项目一开始是没有加Mybatisplus逻辑删除标识的,后面你在加逻辑删除标识时,不是说加了就好了。

你还需要考虑对整体项目有没有影响,如果之前是用updateById做逻辑删除,那就会导致之前的删除失败甚至是报错,这一点一定要注意。

本人有踩过坑!



声明: 公众号如需转载该篇文章,发表文章的头部一定要 告知是转至公众号: 后端元宇宙。同时也可以问本人要markdown原稿和原图片。其它情况一律禁止转载!

原文链接:https://www.cnblogs.com/qdhxhz/p/17282645.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号