• Mybatis-Plus同时使用逻辑删除和唯一索引的问题及解决办法


    1 问题背景

    在开发中,我们经常会有逻辑删除和唯一索引同时使用的情况。但当使用mybatis plus时,如果同时使用逻辑删除和唯一索引,会报数据重复Duplicate entry的问题。

    举例来说,有表user,建立唯一索引(user_name,is_del)

    CREATE TABLE `user` (
      `id` bigint(11) unsigned NOT NULL AUTO_INCREMENT COMMENT 'Id',
      `user_name` varchar(64) DEFAULT NULL COMMENT '用户名',
      `is_del` bigint(13) DEFAULT '0' COMMENT '逻辑删除标识',
      PRIMARY KEY (`id`),
      UNIQUE KEY `unique_user_name` (`user_name`)
    ) ENGINE=InnoDB AUTO_INCREMENT=11 DEFAULT CHARSET=utf8mb4;
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    如果我们插入一条数据user_name='张三’的数据,然后再删除它,这时数据表中存在一条记录,如下图:

    在这里插入图片描述
    这时如果再插入一条’张三’,虽然之前的一条记录已经被逻辑删除,但是唯一索引只建在了user_name上,所以这时会报数据重复的错误

    2 第一次改进

    我们把唯一索引的组合增加is_del字段

    UNIQUE KEY `unique_user_name_is_del` (`user_name`,`is_del`)
    
    • 1

    这下可以再次插入’张三’这条数据,插入后如下图

    在这里插入图片描述
    但是如果第二次删除’张三’,则还是会报错,因为已经有一条[‘张三’,1]的数据,当程序想把另一条’zhangsan’的is_del字段值为1时,会因为数据重复失败!

    3 第二次改进

    此时应该如何改进呢,可以在user表中增加一个del_version字段,用来把已经删除的数据加上版本号,然后将这个字段也加入唯一索引中

    CREATE TABLE `user` (
      `id` bigint(11) unsigned NOT NULL AUTO_INCREMENT COMMENT 'Id',
      `user_name` varchar(64) DEFAULT NULL COMMENT '用户名',
      `del_version` bigint(11) DEFAULT '0' COMMENT '版本标识,用于逻辑删除',
      `is_del` bigint(13) DEFAULT '0' COMMENT '逻辑删除标识',
      PRIMARY KEY (`id`),
      UNIQUE KEY `unique_user_name_is_del_del_version` (`user_name`,`is_del`,`del_version`)
    ) ENGINE=InnoDB AUTO_INCREMENT=11 DEFAULT CHARSET=utf8mb4;
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    未删除的数据del_version=0,已删除的数据del_version修改成这条记录的id(自增id全局唯一),这样既可以保证无法多次插入同名的数据,又可以满足数据可以多次删除的情况
    例如,我们两次删除同样数据后,再重新插入,这时数据表中的数据如下:

    在这里插入图片描述

    4 代码解决方案

    我们使用mybatis plus提供的工具生成代码,这时所有的service层接口都会继承 IService 这个接口,这个接口有很多默认方法,实现了对数据库的操作。

    我们的思路是,新建一个IBaseService接口,继承IService接口。在这个IBaseService接口中,重写所有和删除相关的方法,在其中设置【del_version】=【自增id】。而原来的所有service层接口,不再继承IService,而是继承我们新的IBaseService。

    这样就解决了逻辑删除和唯一索引共用的问题,IBaseService具体代码如下:

    import com.baomidou.mybatisplus.core.conditions.Wrapper;
    import com.baomidou.mybatisplus.core.toolkit.CollectionUtils;
    import com.baomidou.mybatisplus.extension.service.IService;
    import com.baomidou.mybatisplus.extension.toolkit.SqlHelper;
    import org.llbqhh.dao.entity.BaseDO;
     
    import java.io.Serializable;
    import java.util.Collection;
    import java.util.List;
    import java.util.Map;
    import java.util.Objects;
     
    /**
     * @Author wuKeFan
     * @Date 2023/11/08
     * @Description: 逻辑删除前先更新版本号
     */
    public interface IBaseService<T extends BaseDO> extends IService<T> {
        /**
         * 根据 ID 删除
         *
         * @param id 主键ID
         */
        @Override
        default boolean removeById(Serializable id) {
            T objDO = getBaseMapper().selectById(id);
            return beforeDelete(objDO) && SqlHelper.retBool(getBaseMapper().deleteById(id));
        }
     
        /**
         * 删除对象前,先修改其版本号
         * @param objDO
         * @return
         */
        default boolean beforeDelete(T objDO) {
            if (Objects.isNull(objDO)) {
                return false;
            }
            // 逻辑删除前先更新版本号
            objDO.setDelVersion(objDO.getId());
            return SqlHelper.retBool(getBaseMapper().updateById(objDO));
        }
     
        /**
         * 根据 columnMap 条件,删除记录
         *
         * @param columnMap 表字段 map 对象
         */
        @Override
        default boolean removeByMap(Map<String, Object> columnMap) {
            throw new RuntimeException("不支持的数据库删除操作");
        }
     
        /**
         * 根据 entity 条件,删除记录
         *
         * @param queryWrapper 实体包装类 {@link com.baomidou.mybatisplus.core.conditions.query.QueryWrapper}
         */
        @Override
        default boolean remove(Wrapper<T> queryWrapper) {
            List<T> objDOS = getBaseMapper().selectList(queryWrapper);
            if (CollectionUtils.isNotEmpty(objDOS)) {
                objDOS.forEach(objDO -> beforeDelete(objDO));
            }
            return SqlHelper.retBool(getBaseMapper().delete(queryWrapper));
        }
     
        /**
         * 删除(根据ID 批量删除)
         *
         * @param idList 主键ID列表
         */
        @Override
        default boolean removeByIds(Collection<? extends Serializable> idList) {
            if (CollectionUtils.isEmpty(idList)) {
                return false;
            }
            List<T> objDOS = getBaseMapper().selectBatchIds(idList);
            if (CollectionUtils.isNotEmpty(objDOS)) {
                objDOS.forEach(objDO -> beforeDelete(objDO));
            }
            return SqlHelper.retBool(getBaseMapper().deleteBatchIds(idList));
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
  • 相关阅读:
    YashanDB混合存储揭秘:行式存储如何为高效TP业务保驾护航(上)
    Java虚拟机(JVM)面试专题(初级程序员P6)
    美赞臣EDI 940仓库装运订单详解
    Quartz.Net 主要概念介绍和吐槽
    CUDA生态系统架构是什么样的?CUDA的技术原理是什么?底层原理是什么?怎么开发相关产品
    云里黑白第十九回——我们无法判断你的电脑是否已准备好继续安装Windows 10
    III.二分算法
    「Verilog学习笔记」实现3-8译码器①
    java泛型入门篇
    Qt应用开发(基础篇)——视图基类 QAbstractItemView
  • 原文地址:https://blog.csdn.net/qq_37284798/article/details/134280999