我的编程空间,编程开发者的网络收藏夹
学习永远不晚

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

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

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

需求背景

比如有张用户表,在插入或者更新数据的时候,我们需要 用户名称(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是支持逻辑删除的,如果确定在哪个字段是逻辑删除字段,那就在该字段上添加一个注解

  
    @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;

    
    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做逻辑删除,那就会导致之前的删除失败甚至是报错,这一点一定要注意。

到此这篇关于MyBatisPlus解决逻辑删除与唯一索引的兼容问题的文章就介绍到这了,更多相关MyBatisPlus逻辑删除与唯一索引内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

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

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

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

这篇文章主要介绍了MyBatisPlus解决逻辑删除与唯一索引的兼容问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
2023-05-15

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

这篇文章主要介绍“怎么用MyBatisPlus解决逻辑删除与唯一索引的兼容问题”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“怎么用MyBatisPlus解决逻辑删除与唯一索引的兼容问题”文章能帮助大
2023-07-06

MyBatis-Plus解决逻辑删除与唯一索引的问题

本文主要介绍了MyBatis-Plus解决逻辑删除与唯一索引的问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2022-11-13

MyBatisPlus逻辑删除与唯一索引冲突的示例分析

小编给大家分享一下MyBatisPlus逻辑删除与唯一索引冲突的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!1、问题背景:在开发中,我们经常会有逻辑删除
2023-06-14

mybatisplus的逻辑删除问题怎么解决

本文小编为大家详细介绍“mybatisplus的逻辑删除问题怎么解决”,内容详细,步骤清晰,细节处理妥当,希望这篇“mybatisplus的逻辑删除问题怎么解决”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。myb
2023-07-05

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录