详解MyBatisPlus逻辑删除与唯一索引冲突问题

1、问题背景:

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

举个例子:

原来数据库结构:

这里location_id是唯一索引

CREATE TABLE `eam_location` (

`id` int(11) NOT NULL AUTO_INCREMENT,

`location_id` varchar(50) UNIQUE NOT NULL COMMENT '位置代码',

`location_level` tinyint(1) NOT NULL COMMENT '位置级别',

`location_name` varchar(50) NOT NULL COMMENT '位置名称',

`parent_location_id` varchar(50) COMMENT '上级位置代码',

`delete_flag` tinyint(1) DEFAULT 0 COMMENT '软删除',

`version` int(11) DEFAULT 1 COMMENT '乐观锁',

PRIMARY KEY (`id`)

) ENGINE=InnoDB DEFAULT CHARSET=utf8;

这里在删除添加某一条数据后,delete_flag变成0,当删除后delete_flag会变成1,再次添加相同的数据时,由于数据库检测不到原来数据,会报数据重复Duplicate entry的问题

解决办法:参考逻辑删除与唯一约束的需求冲突

SQL数据结构,将delete_flag用时间戳进行表示,唯一索引变成了联合唯一索引 UNIQUE KEY unique_location_delete_flag(location_id, delete_flag) ,当添加一条数据时,delete_flag变成null,当删除数据时,delete_flag变成删除时的一个时间戳。再次添加相同数据时,由于添加的数据是联合唯一索引unique_location_delete_flag ,delete_flag为null,不会产生冲突,多次删除也是,完美解决问题。

CREATE TABLE `eam_location` (

`id` int(11) NOT NULL AUTO_INCREMENT,

`location_id` varchar(50) NOT NULL COMMENT '位置代码',

`location_level` tinyint(1) NOT NULL COMMENT '位置级别',

`location_name` varchar(50) NOT NULL COMMENT '位置名称',

`parent_location_id` varchar(50) COMMENT '上级位置代码',

`delete_flag` datetime COMMENT '软删除',

`version` int(11) DEFAULT 1 COMMENT '乐观锁',

PRIMARY KEY (`id`),

UNIQUE KEY `unique_location_delete_flag`(`location_id`, `delete_flag`)

) ENGINE=InnoDB DEFAULT CHARSET=utf8;

entity类:

@AllArgsConstructor

@NoArgsConstructor

@Builder(toBuilder = true)

@Data

@EqualsAndHashCode(callSuper = false)

public class EamEquipmentType implements Serializable {

private static final long serialVersionUID = 1L;

/**

* 数据库自增id

*/

@TableId(value = "id", type = IdType.AUTO)

private Integer id;

/**

* 设备类型编号

*/

private String typeId;

/**

* 设备类型

*/

private String typeName;

/**

* 设备厂商

*/

private String manufacture;

/**

* 设备型号

*/

private String model;

/**

* 标准设备bom 0:未创建 1:已创建

*/

private Boolean typeBom;

/**

* 标准设备bom id

*/

private Integer typeBomId;

/**

* 创建时间

*/

private LocalDateTime createTime;

/**

* 软删除

*/

@TableLogic()

@TableField(fill = FieldFill.INSERT)

private LocalDateTime deleteFlag;

/**

* 乐观锁

*/

@Version

@TableField(fill = FieldFill.INSERT)

private Integer version;

yml配置文件:

mybatis-plus:

global-config:

db-config:

logic-delete-value: "now()" #逻辑删除值是个db获取时间的函数

logic-not-delete-value: "null" #逻辑未删除值为字符串 "null"

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们[email protected] 处理,核实后本网站将在24小时内删除侵权内容。