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

news2025/1/23 3:02:45

需求背景

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

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

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-sql"><span style="color:#f92672">create</span> <span style="color:#f92672">unique</span> index uni_username <span style="color:#f92672">on</span> <span style="color:#f92672">user</span>(username)
</code></span></span>

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

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

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

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-sql"><span style="color:#f92672">create</span> <span style="color:#f92672">unique</span> index uni_username_deleted <span style="color:#f92672">on</span> <span style="color:#f92672">user</span>(username,deleted)
</code></span></span>

这样就ok了吗?

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

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

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

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

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

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-undefined">【强制】业务上具有唯一特性的字段,即使是组合字段,也必须建成唯一索引。
</code></span></span>

手册还有这么一句话:

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-undefined">即使在应用层做了非常完善的校验和控制,只要没有唯一索引,根据墨菲定律,必然有脏数据产生。
</code></span></span>

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

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

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

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-bash">我们依旧可以将 username + deleted 组合成一个联合唯一索引,但是删除的时候deleted不再是固定的1,而是当前的主键ID,也就是deleted不等于0都是删除状态,如果删除了那deleted值=<span style="color:#e6db74">id</span>值
</code></span></span>

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

二、MyBatisPlus逻辑删除

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

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-java">  <span style="color:#75715e">/**
     * 1、删除 0、未删除
     */</span>
    <span style="color:#75715e">@TableLogic(value = "0", delval = "1")</span>
    <span style="color:#f92672">private</span> Integer deleted;
</code></span></span>

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

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

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

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-java">    <span style="color:#75715e">@TableLogic(value = "0", delval = "id")</span>
    <span style="color:#f92672">private</span> Integer deleted;
</code></span></span>

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

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-yaml language-yml"><span style="color:#f92672">mybatis-plus:</span>
  <span style="color:#f92672">global-config:</span>
    <span style="color:#f92672">db-config:</span>
      <span style="color:#f92672">logic-delete-value:</span> <span style="color:#ae81ff">1</span> <span style="color:#75715e"># 逻辑已删除值(默认为 1)</span>
      <span style="color:#f92672">logic-not-delete-value:</span> <span style="color:#ae81ff">0</span> <span style="color:#75715e"># 逻辑未删除值(默认为 0)</span>
</code></span></span>

三、测试

1、用户表

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-sql"><span style="color:#f92672">CREATE</span> <span style="color:#f92672">TABLE</span> `<span style="color:#f92672">user</span>` (
  `id` <span style="color:#e6db74">int</span> unsigned  AUTO_INCREMENT COMMENT <span style="color:#e6db74">'主键'</span>,
  `username` <span style="color:#e6db74">varchar</span>(<span style="color:#ae81ff">128</span>)  COMMENT <span style="color:#e6db74">'用户名'</span>,
  `phone` <span style="color:#e6db74">varchar</span>(<span style="color:#ae81ff">32</span>)  COMMENT <span style="color:#e6db74">'手机号'</span>,
  `sex` <span style="color:#e6db74">char</span>(<span style="color:#ae81ff">1</span>)  COMMENT <span style="color:#e6db74">'性别'</span>,
  `create_time` datetime  COMMENT <span style="color:#e6db74">'创建时间'</span>,
  `update_time` datetime  COMMENT <span style="color:#e6db74">'更新时间'</span>,
  `deleted` tinyint <span style="color:#f92672">DEFAULT</span> <span style="color:#e6db74">'0'</span> COMMENT <span style="color:#e6db74">'1、删除 0、未删除'</span>,
  <span style="color:#f92672">PRIMARY</span> KEY (`id`)
) ENGINE<span style="color:#ab5656">=</span>InnoDB AUTO_INCREMENT<span style="color:#ab5656">=</span><span style="color:#ae81ff">1</span> 
</code></span></span>

2、创建对应实体

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-java"><span style="color:#75715e">@Data</span>
<span style="color:#75715e">@Accessors(chain = true)</span>
<span style="color:#75715e">@TableName("user")</span>
<span style="color:#f92672">public</span> <span style="color:#f92672">class</span> <span style="color:#a6e22e">UserDO</span> <span style="color:#f92672">implements</span> <span style="color:#a6e22e">Serializable</span> {

    <span style="color:#f92672">private</span> <span style="color:#f92672">static</span> <span style="color:#f92672">final</span> <span style="color:#e6db74">long</span> <span style="color:#e6db74">serialVersionUID</span> <span style="color:#ab5656">=</span> <span style="color:#ae81ff">1L</span>;

    <span style="color:#75715e">@TableId(type = IdType.AUTO)</span>
    <span style="color:#f92672">private</span> Integer id;
    <span style="color:#75715e">/**
     * 用户名
     */</span>
    <span style="color:#f92672">private</span> String username;
    <span style="color:#75715e">/**
     * 手机号
     */</span>
    <span style="color:#f92672">private</span> String phone;
    <span style="color:#75715e">/**
     * 性别
     */</span>
    <span style="color:#f92672">private</span> String sex;
    <span style="color:#75715e">/**
     * 创建时间
     */</span>
    <span style="color:#f92672">private</span> LocalDateTime createTime;
    <span style="color:#75715e">/**
     * 更新时间
     */</span>
    <span style="color:#f92672">private</span> LocalDateTime updateTime;

    <span style="color:#75715e">/**
     * 1、删除 0、未删除
     */</span>
    <span style="color:#f92672">private</span> Integer deleted;
}
</code></span></span>

3、物理删除测试

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

我们来看下删除示例

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-java">    <span style="color:#75715e">@Test</span>
    <span style="color:#f92672">public</span> <span style="color:#f92672">void</span> <span style="color:#a6e22e">deleteById</span><span style="color:#f8f8f2">()</span> {
        <span style="color:#75715e">//方式一:根据id删除</span>
        mapper.deleteById(<span style="color:#ae81ff">10</span>);
        <span style="color:#75715e">//方式二:根据指定字段删除</span>
        LambdaQueryWrapper<UserDO> wrapper = <span style="color:#f92672">new</span> <span style="color:#a6e22e">LambdaQueryWrapper</span><>();
        wrapper.eq(UserDO::getSex, <span style="color:#e6db74">"男"</span>);
        mapper.delete(wrapper);
        <span style="color:#75715e">//方式三:手动逻辑删除</span>
        <span style="color:#e6db74">UserDO</span> <span style="color:#e6db74">userDO</span> <span style="color:#ab5656">=</span> <span style="color:#f92672">new</span> <span style="color:#a6e22e">UserDO</span>();
        userDO.setId(<span style="color:#ae81ff">10</span>);
        userDO.setDeleted(<span style="color:#ae81ff">1</span>);
        mapper.updateById(userDO);
    }
</code></span></span>

执行结果

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-sql"><span style="color:#75715e">--方式1</span>
<span style="color:#f92672">DELETE</span> <span style="color:#f92672">FROM</span> <span style="color:#f92672">user</span> <span style="color:#f92672">WHERE</span> id<span style="color:#ab5656">=</span><span style="color:#ae81ff">10</span>
<span style="color:#75715e">--方式2</span>
<span style="color:#f92672">DELETE</span> <span style="color:#f92672">FROM</span> <span style="color:#f92672">user</span> <span style="color:#f92672">WHERE</span> (sex <span style="color:#ab5656">=</span> <span style="color:#e6db74">'男'</span>)
<span style="color:#75715e">--方式3</span>
<span style="color:#f92672">UPDATE</span> <span style="color:#f92672">user</span> <span style="color:#f92672">SET</span> deleted<span style="color:#ab5656">=</span><span style="color:#ae81ff">1</span> <span style="color:#f92672">WHERE</span> id<span style="color:#ab5656">=</span><span style="color:#ae81ff">10</span>
</code></span></span>

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

4、逻辑删除测试

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

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-java"> <span style="color:#75715e">@TableLogic(value = "0", delval = "id")</span>
 <span style="color:#f92672">private</span> Integer deleted;
</code></span></span>

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

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-sql"><span style="color:#75715e">--方式1</span>
<span style="color:#f92672">UPDATE</span> <span style="color:#f92672">user</span> <span style="color:#f92672">SET</span> deleted<span style="color:#ab5656">=</span>id <span style="color:#f92672">WHERE</span> id<span style="color:#ab5656">=</span><span style="color:#ae81ff">10</span> <span style="color:#f92672">AND</span> deleted<span style="color:#ab5656">=</span><span style="color:#ae81ff">0</span>
<span style="color:#75715e">--方式2</span>
<span style="color:#f92672">UPDATE</span> <span style="color:#f92672">user</span> <span style="color:#f92672">SET</span> deleted<span style="color:#ab5656">=</span>id <span style="color:#f92672">WHERE</span> deleted<span style="color:#ab5656">=</span><span style="color:#ae81ff">0</span> <span style="color:#f92672">AND</span> (sex <span style="color:#ab5656">=</span> <span style="color:#e6db74">'男'</span>)
<span style="color:#75715e">--方式3</span>
报错了
</code></span></span>

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

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

 

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

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-sql"><span style="color:#f92672">UPDATE</span> <span style="color:#f92672">user</span>  <span style="color:#f92672">WHERE</span> id<span style="color:#ab5656">=</span>?  <span style="color:#f92672">AND</span> deleted<span style="color:#ab5656">=</span><span style="color:#ae81ff">0</span>
</code></span></span>

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

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-sql"><span style="color:#f92672">UPDATE</span> <span style="color:#f92672">user</span> <span style="color:#f92672">SET</span> deleted<span style="color:#ab5656">=</span><span style="color:#ae81ff">1</span>  <span style="color:#f92672">WHERE</span> id<span style="color:#ab5656">=</span>?  <span style="color:#f92672">AND</span> deleted<span style="color:#ab5656">=</span><span style="color:#ae81ff">0</span>
</code></span></span>

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

 

真相大白了

<span style="color:#4b4b4b"><span style="background-color:#ffffff"><code class="language-sql">Mybatisplus在updateById更新时,如果已经加了逻辑删除标记,那做<span style="color:#f92672">SQL</span>拼接的时候,会自动过滤掉逻辑删除的<span style="color:#f92672">Set</span>拼接
</code></span></span>

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

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

本人有踩过坑!  

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/852115.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

JavaWeb-能制作中小型网站企业管理系统(适合快速梳理)

导言 第一章 Web前端开发 一、初始web前端 二、HTML、CSS介绍 三、VsCode安装 四、标签 1 实现标题--排版 2 实现标题--样式 颜色表示 CSS选择器 3 实现标题--超链接 4 实现标签--正文 5 页面布局 盒子模型 布局标签 6 表格、表单标签 表格标签 表单标签 表单项 …

Electron学习1 安装环境与第一个程序

Electron学习1 安装环境与第一个程序 一、 Electron 简介二、安装 nvm三、安装nodejs四、安装nrm五、安装electron1. npm 初始化2. 创建 package.json3. 安装electron4. 创建一个页面5. 创建文件main.js6. 创建预加载器文件 preload.js7. 启动程序 六、打包 一、 Electron 简介…

大模型在金融医疗、生命系统和物理仿真领域的创新应用探索

点击蓝字 关注我们 AI TIME欢迎每一位AI爱好者的加入&#xff01; 在当今迅速发展的科技领域&#xff0c;大模型技术正日益成为金融医疗、生命系统和物理仿真等领域中的重要工具。2023年6月16日&#xff0c;AI TIME举办的青年科学家大模型专场活动邀请了国防科技大学理学院数学…

springboot+vue智能化网络电子相册图片管理系统_84ds3

随着计算机技术发展&#xff0c;计算机系统的应用已延伸到社会的各个领域&#xff0c;大量基于网络的广泛应用给生活带来了十分的便利。所以把智能化电子相册与现在网络相结合&#xff0c;利用计算机搭建智能化电子相册系统&#xff0c;实现智能化电子相册的信息化。则对于进一…

超级浏览器与指纹浏览器:功能与特点的比较

导语&#xff1a;随着互联网的快速发展&#xff0c;隐私和安全问题日益受到关注。在这个背景下&#xff0c;超级浏览器和指纹浏览器作为定制化浏览器的两个重要类型&#xff0c;各自具有独特的功能和特点。本文将对超级浏览器和指纹浏览器进行比较&#xff0c;帮助读者更好地理…

微信小程序nodejs+vue+uniapp校运会高校运动会报名管理系统

3.1小程序端 小程序登录页面&#xff0c;用户也可以在此页面进行注册并且登录等。 登录成功后可以在我的个人中心查看自己的个人信息或者修改信息等 在广播信息中我们可以查看校运会发布的一些信息情况。 在首页我们可以看到校运会具体有什么项目运动。 在查看具体有什么活动我…

linux I/O性能优化

Linux 文件系统 磁盘和文件系统的关系&#xff1a; 磁盘为系统提供了最基本的持久化存储。 文件系统则在磁盘的基础上&#xff0c;提供了一个用来管理文件的树状结构。 文件系统工作原理 索引节点和目录项 文件系统&#xff0c;本身是对存储设备上的文件&#xff0c;进行组织…

【腾讯云 Cloud Studio 实战训练营】使用 Cloud Studio 快速构建 Vue + Vite 完成律师 H5 页面

【腾讯云 Cloud Studio 实战训练营】使用 Cloud Studio 快速构建 Vue Vite 完成律师 H5 页面 前言一、基本介绍1.应用场景2.产品优势 二、准备工作1.注册 Cloud Studio2.进入 Vue 预置开发环境 三、使用 Cloud Studio 快速构建 Vue Vite 完成律师 H5 页面1.安装相关依赖包2.主…

【FAQ】安防监控视频EasyCVR平台分发的FLV视频流在VLC中无法播放

众所周知&#xff0c;TSINGSEE青犀视频汇聚平台EasyCVR可支持多协议方式接入&#xff0c;包括主流标准协议国标GB28181、RTSP/Onvif、RTMP等&#xff0c;以及厂家私有协议与SDK接入&#xff0c;包括海康Ehome、海大宇等设备的SDK等。在视频流的处理与分发上&#xff0c;视频监控…

C语言---数据结构实验---哈夫曼树及哈夫曼编码的算法实现---图的基本操作

文章目录 写在前面哈夫曼树及哈夫曼编码的算法实现实验内容代码实现 图的基本操作实验内容代码实现 写在前面 本篇实验代码非本人写&#xff0c;代码源自外部&#xff0c;经调试解决了部分warning和error后在本地vs上可以正常运行&#xff0c;如有运行失败可换至vs 未来会重构…

将朴素矩阵乘法在共享内存中分块,每个线程只计算结果矩阵中的单个元素

kenel的block中的每个线程用于计算共享内存中矩阵Pd中的一个元素Pd_(i&#xff0c;j)&#xff0c;每个线程都读取Md的一行和Nd的一列。Pd_(0&#xff0c;0)和Pd_(1,0)两个结果是由两个线程完成的。这里一开始只有Pd被加载进共享内存&#xff0c;Md和Nd还在全局内存中&#xff1…

嵌入式软件测试方法-质量模型

软件测试评估质量的时候用到的很多测试度量项 质量大师朱兰提出了“质量管理三部曲”&#xff0c;来对企业质量进行管理。 第一部曲&#xff1a;质量策划&#xff0c;致力于制定质量目标并规定必要的运行过程、准备相关资源以实现质量目标。 第二部曲&#xff1a;质量控制&am…

【java】【经验】java: 错误: 不支持发行版本 6

前言&#xff1a;配置过maven之后&#xff0c;发现原来的一些项目运行提示java: 错误: 不支持发行版本 6或者java: 错误: 不支持发行版本 5&#xff0c;主要原因&#xff1a;是因为项目使用的Java版本和安装的Java版本不符合 目录 1 设置项目java版本 2 设置模块版本 3 set…

马尔可夫链的性质和例子

马尔可夫链的重要性质以及两个例题如下&#xff1a; 注意&#xff1a; 例5中有几个地方需要注意&#xff1a; &#xff08;1&#xff09;为什么 P 11 P 22 p q ( 1 − p ) ( 1 − q ) &#xff0c;而 P 33 p q ( 1 − p ) P_{11} P_{22} pq (1-p)(1-q)&#xff0c;而P…

低代码开发平台:无限潜力,适用于各类应用程序开发!

随着技术的不断进步和市场需求的变化&#xff0c;低代码开发平台成为了构建应用程序的一种热门选择。低代码开发平台通过简化应用程序开发过程&#xff0c;降低了编程门槛&#xff0c;使非技术人员也能够快速构建功能强大的应用程序。不过&#xff0c;低代码开发平台究竟可以开…

[vue-element-admin]下载与安装

一、环境搭建 1 nodejs 源码地址 sudo apt install build-essential # 内含gcc g make等全家桶git clone git://github.com/nodejs/node.git # 下载源码 cd node sudo ./config sudo make && make install # 编译 node -v # 查看是否编译成功二、遇见的问题 问题…

清风数学建模——插值算法

插值法 文章目录 插值法作用定义概念一维插值问题一维插值多项式原理定理 拉格朗日插值法和牛顿插值法埃尔米特插值分段线性插值分段三次埃尔米特插值法代码三次样条插值及其代码例子n维数据的插值&#xff08;了解&#xff09; 作用 数模比赛中&#xff0c;常常需要根据已知的…

QT笔记——QT自定义事件

我们有时候想发送自定义事件 1&#xff1a;创建自定义事件&#xff0c;首先我们需要知道它的条件 1&#xff1a;自定义事件需要继承QEvent 2&#xff1a;事件的类型需要在 QEvent::User 和 QEvent::MaxUser 范围之间&#xff0c;在QEvent::User之前 是预留给系统的事件 3&#…

[ubuntu]创建root权限的用户

一、创建新用户 1、创建新用户 sudo useradd -r -m -s /bin/bash 用户名 # -r&#xff1a;建立系统账号 -m&#xff1a;自动建立用户的登入目录 -s&#xff1a;指定用户登入后所使用的shell2、手动为用户设置密码 passwd 用户名 二、为用户增加root权限 1、添加写权限 ch…

适配器模式:将不兼容的接口转换为可兼容的接口

适配器模式&#xff1a;将不兼容的接口转换为可兼容的接口 什么是适配器模式&#xff1f; 适配器模式是一种结构型设计模式&#xff0c;用于将一个类的接口转换为客户端所期望的另一个接口。它允许不兼容的类能够合作&#xff0c;使得原本由于接口不匹配而无法工作的类能够一…