MyBatis-Plus 逻辑删除实现

news/2025/3/5 13:58:26/

在很多企业级应用中,数据删除操作通常采用 逻辑删除 的方式,而不是物理删除。逻辑删除指的是通过更新字段(例如 is_deletedstatus)来标记数据为删除状态,而不是真的从数据库中删除记录。这样做的好处是保留数据的历史记录,方便后续的数据恢复、审计以及统计。

MyBatis-Plus(MP)提供了强大的 逻辑删除功能,使得开发者可以快速实现这一功能。本文将介绍 MyBatis-Plus 如何实现逻辑删除,包括自动填充字段、配置与实现步骤、常见应用场景,并展示如何使用 remove 方法进行逻辑删除,而不是手动通过 update 来修改删除标记。


1. 逻辑删除的必要性

1.1 逻辑删除的定义

逻辑删除 是指通过修改某个字段(如 is_deleted)的值,将记录标记为已删除,而不是将其从数据库中删除。这种方式不仅保留了数据的完整性,还能避免因误操作导致的数据丢失。

1.2 逻辑删除的优点

  • 数据恢复:可以在数据标记为删除后,依然保留数据,便于后期的恢复。
  • 历史审计:可以查看被删除记录的历史数据,满足合规性要求。
  • 避免数据丢失:比物理删除更安全,可以减少误操作带来的影响。

1.3 适用场景

  • 用户账号删除:用户删除账户时,使用逻辑删除标记账户为已删除,便于后续恢复。
  • 订单或产品状态管理:删除订单或产品记录时,使用逻辑删除来标记记录而非物理删除。
  • 内容管理系统(CMS):在 CMS 中删除文章或评论时,使用逻辑删除来保留数据。

2. MyBatis-Plus 支持的逻辑删除方式

MyBatis-Plus 提供了非常方便的方式来实现逻辑删除,主要依赖于 @TableLogic 注解,标记字段为逻辑删除字段。MyBatis-Plus 会自动处理该字段的值,并在查询时自动排除已删除的记录。

2.1 使用 @TableLogic 注解进行逻辑删除

MyBatis-Plus 提供的 @TableLogic 注解可以标识一个字段为逻辑删除字段。默认情况下,字段值为 0 表示未删除,值为 1 表示已删除。

示例:User 实体类配置
import com.baomidou.mybatisplus.annotation.TableLogic;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.IdType;public class User {@TableId(value = "id", type = IdType.AUTO)private Long id;private String name;private Integer age;@TableLogic  // 逻辑删除字段private Integer isDeleted;// Getters and Setters
}

在这个示例中:

  • @TableLogic:标识 isDeleted 字段为逻辑删除字段。字段值为 0 表示未删除,值为 1 表示已删除。

2.2 配置逻辑删除字段的值

MyBatis-Plus 默认将 isDeleted 字段值为 0 表示未删除,1 表示已删除。开发者可以自定义逻辑删除的字段值。

自定义逻辑删除值
@Configuration
public class MybatisPlusConfig {@Beanpublic MybatisPlusPropertiesCustomizer mybatisPlusPropertiesCustomizer() {return properties -> {properties.getGlobalConfig().setLogicDeleteValue("1");  // 设置逻辑删除的值为 1properties.getGlobalConfig().setLogicNotDeleteValue("0");  // 设置未删除的值为 0};}
}

在这个配置中,设置了逻辑删除字段的值为 1 表示已删除,0 表示未删除。


3. 使用 remove 方法进行逻辑删除

3.1 remove 方法与逻辑删除

MyBatis-Plus 中的 remove 方法用于删除记录,默认情况下会执行物理删除操作。为了启用 逻辑删除,需要通过设置 @TableLogic 注解,并在执行删除操作时,MyBatis-Plus 会自动更新 isDeleted 字段的值,而不是直接删除记录。

示例:remove 方法进行逻辑删除
public void logicDeleteUser(Long userId) {userService.removeById(userId);  // 使用 remove 方法进行逻辑删除
}

在这个例子中,removeById 方法会自动更新 isDeleted 字段,而不是直接从数据库中删除记录。

3.2 removeById 的原理

removeById 方法的实现是基于 @TableLogic 注解的字段,通过更新 isDeleted 字段的值来实现逻辑删除。MyBatis-Plus 在执行删除操作时,会自动生成如下 SQL:

UPDATE user SET is_deleted = 1 WHERE id = ?;

该操作将 is_deleted 字段的值更新为 1,表示该记录已被删除,而不是从数据库中删除该记录。

3.3 查询时排除逻辑删除的记录

MyBatis-Plus 默认会在查询时自动排除逻辑删除的记录,即 isDeleted 字段为 1 的记录不会被返回。

查询时的自动排除
QueryWrapper<User> queryWrapper = new QueryWrapper<>();
queryWrapper.eq("name", "John");
List<User> users = userService.list(queryWrapper);  // 自动排除逻辑删除的记录

在这个查询中,MyBatis-Plus 会自动忽略 isDeleted = 1 的记录,只返回未删除的记录。


4. 示例:逻辑删除字段与查询

4.1 插入数据时的逻辑删除

在插入数据时,isDeleted 字段会默认设置为 0,表示数据未删除。

User user = new User();
user.setName("John");
user.setAge(30);
userService.save(user);  // 插入时,isDeleted 默认为 0

4.2 更新数据时的逻辑删除

在更新数据时,可以通过 removeById 方法将某条记录的 isDeleted 字段更新为 1,表示数据已被逻辑删除。

User user = userService.getById(1L);
user.setIsDeleted(1);  // 设置为已删除
userService.updateById(user);  // 执行更新操作,实际是逻辑删除

4.3 查询时排除逻辑删除的数据

查询时,MyBatis-Plus 会自动过滤掉逻辑删除的记录。你无需显式地排除已删除记录。

QueryWrapper<User> queryWrapper = new QueryWrapper<>();
queryWrapper.eq("name", "John");
List<User> users = userService.list(queryWrapper);  // 自动排除逻辑删除的记录

4.4 查询已删除的记录

如果需要查询已删除的记录,可以通过 QueryWrapper 手动指定查询条件:

QueryWrapper<User> queryWrapper = new QueryWrapper<>();
queryWrapper.eq("is_deleted", 1);  // 查询已删除的记录
List<User> deletedUsers = userService.list(queryWrapper);

5. 总结

  • 逻辑删除的必要性:逻辑删除可以保留数据历史,避免数据丢失,适用于需要审计、数据恢复的场景。
  • MyBatis-Plus 支持的逻辑删除方式:通过 @TableLogic 注解,MyBatis-Plus 能够在插入、更新和查询时自动处理逻辑删除字段。
  • 实现步骤:在实体类中使用 @TableLogic 注解标识逻辑删除字段,配置自定义删除值,并实现 MetaObjectHandler 来处理字段填充。
  • 使用 remove 方法进行逻辑删除:MyBatis-Plus 的 removeById 方法会自动更新 isDeleted 字段,实现逻辑删除,而不是物理删除数据。

MyBatis-Plus 的逻辑删除功能简化了数据的删除操作,保留了删除记录的历史数据,确保了应用的数据一致性和安全性。通过合理配置和使用 MyBatis-Plus 的逻辑删除功能,开发者能够更加高效地管理数据,并满足业务需求。 🚀


http://www.ppmy.cn/news/1576830.html

相关文章

网线水晶头接法

目录 介绍 排线标准 连接方法 直连互联法 交叉互联法 操作步骤 介绍 网线&#xff1a;双绞线&#xff0c;有4对8条芯线&#xff0c;分别为白绿色、绿色、白橙色、蓝色、白蓝色、橙色、白棕色、棕色。 水晶头&#xff1a;也被称为RJ45连接器&#xff0c;是一种用于网络连接…

【应急响应工具教程】一款自动化分析网络安全应急响应工具--FindAll

1、工具介绍 FindAll 是一款安全团队开发的轻量化蓝队工具&#xff0c;专为应急响应场景设计&#xff0c;主打信息收集与威胁情报联动&#xff0c;尤其适合团队快速排查多台主机的安全风险。同时FindAll采用客户端-服务器&#xff08;CS&#xff09;架构&#xff0c;特别适用于…

AI赋能校园安全:科技助力预防与应对校园霸凌

校园本应是学生快乐学习、健康成长的地方&#xff0c;然而&#xff0c;校园霸凌却成为威胁学生身心健康的隐形“毒瘤”。近年来&#xff0c;随着人工智能&#xff08;AI&#xff09;技术的快速发展&#xff0c;AI在校园安全领域的应用逐渐成为解决校园霸凌问题的新突破口。通过…

YOLOv12 ——基于卷积神经网络的快速推理速度与注意力机制带来的增强性能结合

概述 实时目标检测对于许多实际应用来说已经变得至关重要&#xff0c;而Ultralytics公司开发的YOLO&#xff08;You Only Look Once&#xff0c;只看一次&#xff09;系列一直是最先进的模型系列&#xff0c;在速度和准确性之间提供了稳健的平衡。注意力机制的低效阻碍了它们在…

如何使用 Jenkins 实现 CI/CD 流水线:从零开始搭建自动化部署流程

如何使用 Jenkins 实现 CI/CD 流水线:从零开始搭建自动化部署流程 在软件开发过程中,持续集成(CI)和持续交付(CD)已经成为现代开发和运维的标准实践。随着代码的迭代越来越频繁,传统的手动部署方式不仅低效,而且容易出错。为了提高开发效率和代码质量,Jenkins作为一款…

Deepseek的缺陷

Deepseek什么都略懂&#xff0c;但答案很难让人满意。 内容蜻蜓点水&#xff0c;什么都有点&#xff0c;但浅薄的很。让玩家很无语。 一些软件类的答案貌似比较详细&#xff0c;但距离能用还差很远。 ----------------- 一些问题的答案如下&#xff1a; 《b4》是一款经典的生…

NUMA架构介绍

NUMA 架构详解 NUMA&#xff08;Non-Uniform Memory Access&#xff0c;非统一内存访问&#xff09; 是一种多处理器系统的内存设计架构&#xff0c;旨在解决多处理器系统中内存访问延迟不一致的问题。与传统的 UMA&#xff08;Uniform Memory Access&#xff0c;统一内存访问…

DeepSeek开源周第四弹!DeepSeek开源三剑客:训练效率的“时空魔术师”与“资源管家”全解析

开篇语 AI训练场的效率革命正在悄然爆发——当传统流水线还在“单向龟速”中挣扎&#xff0c;DeepSeek的三把利刃已划破算力困局&#xff1a;DualPipe像手术刀般精准切割时间空洞&#xff0c;将GPU利用率推至极限&#xff1b;EPLB化身智能指挥家&#xff0c;让MoE模型的算力交…