您当前的位置:首页 > 电脑百科 > 程序开发 > 框架

MyBatisPlus逻辑删除与唯一索引的兼容问题

时间:2023-04-04 15:33:59  来源:今日头条  作者:王路飞学Java

需求背景

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

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

create unique index uni_username on user(username)

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

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

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

create unique index uni_username_deleted on user(username,deleted)

这样就ok了吗?

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

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

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

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

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

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

手册还有这么一句话:

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

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

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

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

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

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

二、MyBatisPlus逻辑删除

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

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

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

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

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

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

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

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

三、测试

1、用户表

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

2、创建对应实体

@Data
@Accessors(chain = true)
@TableName("user")
public class UserDO implements Serializable {

    private static final long serialVersionUID = 1L;

    @TableId(type = IdType.AUTO)
    private Integer id;
    /**
     * 用户名
     */
    private String username;
    /**
     * 手机号
     */
    private String phone;
    /**
     * 性别
     */
    private String sex;
    /**
     * 创建时间
     */
    private LocalDateTime createTime;
    /**
     * 更新时间
     */
    private LocalDateTime updateTime;

    /**
     * 1、删除 0、未删除
     */
    private Integer deleted;
}

3、物理删除测试

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

我们来看下删除示例

    @Test
    public void deleteById() {
        //方式一:根据id删除
        mApper.deleteById(10);
        //方式二:根据指定字段删除
        LambdaQueryWrapper<UserDO> wrapper = new LambdaQueryWrapper<>();
        wrapper.eq(UserDO::getSex, "男");
        mapper.delete(wrapper);
        //方式三:手动逻辑删除
        UserDO userDO = new UserDO();
        userDO.setId(10);
        userDO.setDeleted(1);
        mapper.updateById(userDO);
    }

执行结果

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

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

4、逻辑删除测试

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

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

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

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

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

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

 

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

UPDATE user  WHERE id=?  AND deleted=0

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

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

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

 

真相大白了

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

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

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

本人有踩过坑!

原文:
https://mp.weixin.qq.com/s/8QdxfdzPaWm5dL50VW5NKA



Tags:MyBatisPlus   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,不构成投资建议。投资者据此操作,风险自担。如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除。
▌相关推荐
需求背景比如有张用户表,在插入或者更新数据的时候,我们需要 用户名称(username),不能重复。我们首先考虑的是给该字段创建唯一索引create unique index uni_username on use...【详细内容】
2023-04-04  Tags: MyBatisPlus  点击:(0)  评论:(0)  加入收藏
ServiceImpl类是我们进行SQL操作中非常重要的一个类,通过MybatisPlus生成的各个实体类的XXXImpl都会继承ServiceImpl类那里继承全部的方法,那么ServiceImpl类中有哪些方法呢?如...【详细内容】
2022-08-03  Tags: MyBatisPlus  点击:(1037)  评论:(0)  加入收藏
今天介绍一个 MyBatis - Plus 官方发布的神器:mybatis-mate 为 mp 企业级模块,支持分库分表,数据审计、数据敏感词过滤(AC算法),字段加密,字典回写(数据绑定),数据权限,表结构自动生成...【详细内容】
2022-06-17  Tags: MyBatisPlus  点击:(150)  评论:(0)  加入收藏
1. Mybatis 存在的痛点我们知道 MyBatis 是一个基于 java 的持久层框架,它内部封装了 jdbc,极大提高了我们的开发效率。但是使用 Mybatis 开发也有很多痛点: 每个 Dao 接口都需...【详细内容】
2022-06-14  Tags: MyBatisPlus  点击:(192)  评论:(0)  加入收藏
MybatisPlus是国产的第三方插件, 它封装了许多常用的CURDapi,免去了我们写mapper.xml的重复劳动,这里介绍了基本的整合SpringBoot和基础用法。2|0引入依赖在项目中pom文件引入m...【详细内容】
2022-05-05  Tags: MyBatisPlus  点击:(147)  评论:(0)  加入收藏
SpringBoot 集成 MybatisPlus 系列SpringBoot 版本:2.6.4 MybatisPlus 版本:3.5.1 SpringBoot整合MybatisPlus SpringBoot整合MybatisPlus数据自动填充 SpringBoot整合Mybatis...【详细内容】
2022-03-18  Tags: MyBatisPlus  点击:(440)  评论:(0)  加入收藏
一、前言1.1、关于枚举类① 枚举是JDK1.5中的新功能,我们可以使用枚举很好的去描述一些业务场景:一年有四季、人类有男女...② 同样我们在业务层面会有很多,比如状态属性、分...【详细内容】
2022-01-20  Tags: MyBatisPlus  点击:(537)  评论:(0)  加入收藏
(一)前言最早写JDBC的时候,要手动配连接信息,要一条条手写sql语句。后来Mybatis出现了,不需要再手动配置连接信息,sql语句也和代码隔离开来,但是还免不了写Sql。接着出现了MybatisP...【详细内容】
2021-09-17  Tags: MyBatisPlus  点击:(317)  评论:(0)  加入收藏
▌简易百科推荐
概念: Django管理器是Django框架中的一个组件,它提供了一种将Python代码与数据库查询和操作进行交互的方式。每个Django模型都有一个默认的管理器,这个管理器提供了一些基本的...【详细内容】
2023-05-18  你的老师父  今日头条  Tags:Django   点击:(2)  评论:(0)  加入收藏
了解如何在 Kubernetes 上安装 Pinot Control Plane,配置您的第一个 Pinot 集群,并深入了解控制平面功能。作为一名经验丰富的专业人士,我作为 Apache Druid k8s operator 的维...【详细内容】
2023-05-18    科技狠活与软件技术  Tags:Kubernetes   点击:(2)  评论:(0)  加入收藏
什么是自动化测试在软件测试中,自动化测试指的是使用独立于待测软件的其他软件来自动执行测试、比较实际结果与预期并生成测试报告这一过程。在测试流程已经确定后,测试自动化...【详细内容】
2023-05-18    政采云技术  Tags:框架   点击:(1)  评论:(0)  加入收藏
大家好呀,我是楼仔。RabbitMQ 的文章之前写过,但是当时给的示例是 Demo 版的,这篇文章主要是结合之前写的理论知识,将 RabbitMQ 集成到技术派项目中。不 BB,上文章目录:下面我们先...【详细内容】
2023-05-15  楼仔    Tags:RabbitMQ   点击:(11)  评论:(0)  加入收藏
如果你觉得k8s太重了,试试k3s吧。k3s是一个轻量级的Kubernetes发行版,它可以在低资源的环境中运行,比如边缘计算或者物联网设备。k3s包含了Kubernetes的核心功能,但是移除了一些...【详细内容】
2023-05-12  T锅侠  今日头条  Tags:k3s   点击:(1)  评论:(0)  加入收藏
MyBatis 是一个优秀的持久层框架,它提供了丰富的 SQL 映射功能,可以让我们通过 XML 或注解方式来定义 SQL 语句。它很大程度上简化了数据库操作,提高了开发效率。动态 SQL 是...【详细内容】
2023-05-12  互联网高级架构师  稀土掘金  Tags:MyBatis   点击:(18)  评论:(0)  加入收藏
今天分享一下【Kubernetes】 DaemonSet 详解,丰富个人简历,提高面试level,给自己增加一点谈资,秒变面试小达人,BAT不是梦。三分钟你将学会: DaemonSet是什么? DaemonSet的应用场景 ...【详细内容】
2023-05-08  哪吒编程  微信公众号  Tags:K8S   点击:(24)  评论:(0)  加入收藏
前言我们早期在做Spring项目的时候,需要手动添加很多xml配置,比如整合SpringMVC的配置,开启事务,整合Mybatis等等。这些都需要我们手动编写大量的配置,简直就是配置地狱。Spring...【详细内容】
2023-05-08    尚硅谷教育  Tags:SpringBoot   点击:(13)  评论:(0)  加入收藏
一、需求开发过程需求:前端浏览器请求url: http://localhost:8080/demo/handle01,前端⻚面显示后台服务器的时间开发过程1)配置DispatcherServlet前端控制器2)开发处理具体业务逻...【详细内容】
2023-05-06  日拱一卒程序猿  今日头条  Tags:Spring MVC   点击:(15)  评论:(0)  加入收藏
大家好,我是小米,一个热爱技术分享的程序员。今天我们来谈一下Spring事务传播。在使用Spring进行数据库操作时,我们经常会遇到需要使用事务的情况,而Spring对事务的管理非常方便...【详细内容】
2023-05-06  知其然亦知其所以然    Tags:Spring   点击:(20)  评论:(0)  加入收藏
站内最新
站内热门
站内头条