数据库防止误删操作——打数据标记
本文讲解,如何在数据库层面上通过,打数据标记的方法,防止用户的数据误删的情况。
简介
对于数据库删除操作,在很多情况下并不是真正的删除,而是将数据标记为删除状态,以防止误删或者恢复数据。这样做的好处是可以节省物理存储空间,并且避免了数据被永久性删除带来的风险。在实现上也比直接删除更为方便,同时通过备份策略保留历史增量备份,则也可以完整记录历史变化。
逻辑删除:逻辑删除即将数据的删除标记位设置为true或者false。删除标记位的设置只是在数据表中新增一个字段(例如一个布尔型值),用于标识当前的该条数据是否被删除。在查询时通过加上where deleted=false条件进行过滤,从而达到了“删除”的效果。应用程序在更新和查询数据时需要注意,如果应用没有经过精心设计,那么在代码中如果忘记加上deleted=false的条件就会导致出现已被删除的数据无法获取,或者已经删除的数据仍旧能够向外部界面展示。
实现演示
通过这个最为简单的图书管理系统中的一张表,进行讲解
-- ---------------------------- -- Table structure for book -- ---------------------------- DROP TABLE IF EXISTS `book`; CREATE TABLE `book` ( `isbn` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NOT NULL, `title` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NOT NULL, `author` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NOT NULL, `publisher` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NOT NULL, `pubdate` datetime NOT NULL, `price` decimal(10, 2) NOT NULL, `id` int NOT NULL AUTO_INCREMENT, PRIMARY KEY (`id`, `isbn`) USING BTREE ) ENGINE = InnoDB AUTO_INCREMENT = 5 CHARACTER SET = utf8mb4 COLLATE = utf8mb4_0900_ai_ci ROW_FORMAT = DYNAMIC; -- ---------------------------- -- Records of book -- ---------------------------- INSERT INTO `book` VALUES ('9780439227148', 'The Call of the Wild', 'Jack London', 'Scholastic Press', '2001-01-01 00:00:00', 39.40, 1); INSERT INTO `book` VALUES ('9787501592401', 'The Old Man and the Sea', 'Ernest Hemingway', 'Knowledge Press', '2023-01-30 00:00:00', 25.80, 2); INSERT INTO `book` VALUES ('9787501592401', 'The Old Man and the Sea', 'Ernest Hemingway', 'Knowledge Press', '2023-01-31 13:02:42', 25.80, 3); INSERT INTO `book` VALUES ('9780439227148', 'The Call of the Wild', 'Jack London', 'Scholastic Press', '2023-01-30 16:00:00', 34.90, 6); INSERT INTO `book` VALUES ('9781772262902', 'Oliver Twist', 'Charles Dickens', 'Engage Books', '2023-01-03 16:00:00', 45.00, 7);
为了实现逻辑删除,我们可以向表中添加一个标志位(例如:deleted),用于标识记录是否已被删除。如果一条记录被删除,则将标志位设为1。而此时查询操作需要过滤掉被删除的记录。
以下是实现逻辑删除的步骤:
1. 修改数据表
在 book 表中添加一个名为 deleted 的 tinyint 类型的字段,默认值为 0,表示未删除。如下:
ALTER TABLE `book` ADD COLUMN `deleted` TINYINT(1) NOT NULL DEFAULT '0';
注意:MyBatis-Plus 中要求实体类属性名和数据库字段名必须一致或者使用 @TableField 注解标识对应的数据库字段名,因此需要在 Book 实体类中添加 deleted 属性。
2. 定义 BaseMapper
首先定义一个 BaseMapper 接口类,并继承 MyBatis-Plus 所提供的 BaseMapper 接口,添加一个自定义方法 updateDeletedById。updateDeletedById 方法将根据给定 id 设置该条记录的 deleted 值为 1,即标记为已删除。
import com.baomidou.mybatisplus.core.mapper.BaseMapper; import com.example.mybatisdemo.entity.Book; import org.apache.ibatis.annotations.Param; public interface MyBaseMapper<T> extends BaseMapper<T> { /** * 根据id更新删除标志 * * @param id 主键id * @return int */ int updateDeletedById(@Param("id") Integer id); }
3. 定义 BookMapper 接口
接下来,定义一个 BookMapper 接口,并继承 MyBaseMapper 接口。为了满足通用 Curd 操作,可以通过继承 MybatisPlus-BaseMapper 接口实现更多的 CRUD 相关操作。
import com.example.mybatisdemo.mapper.MyBaseMapper; import com.example.mybatisdemo.entity.Book; public interface BookMapper extends MyBaseMapper<Book> { }
4. 定义 BookService 接口和实现类
在 Service 层定义对应的接口并继承 MyBatis-Plus 的 IService 接口,这个接口提供很多基础的 Curd 方法。并在 BookServiceImpl 中实现 updateDeletedById 方法,如下:
BookService.java 文件:
import com.baomidou.mybatisplus.extension.service.IService; import com.example.mybatisdemo.entity.Book; public interface BookService extends IService<Book> { /** * 根据id删除书籍(逻辑删除) * * @param id ID * @return boolean */ boolean removeBookById(Integer id); }
BookServiceImpl.java 文件:
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.example.mybatisdemo.entity.Book; import com.example.mybatisdemo.mapper.BookMapper; import com.example.mybatisdemo.service.BookService; import org.springframework.stereotype.Service; @Service public class BookServiceImpl extends ServiceImpl<BookMapper, Book> implements BookService { /** * 根据id删除书籍(逻辑删除) * * @param id ID * @return boolean 是否删除成功 */ @Override public boolean removeBookById(Integer id) { // 调用 MyBaseMapper 的 updateDeletedById 方法,将对应 id 的记录的 deleted 字段值设置为 1,标记为已删除 int i = baseMapper.updateDeletedById(id); // 如果更新成功返回 true,否则返回 false return i > 0; }
5. 在 Controller 层调用相关方法
在 Controller 中注入 BookService,然后控制层通过调用 Service 层的 removeBookById 方法实现逻辑删除。
import com.example.mybatisdemo.entity.Book; import com.example.mybatisdemo.service.BookService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.*; @RestController @RequestMapping("/book") public class BookController { @Autowired private BookService bookService; @DeleteMapping("/{id}") public boolean deleteBook(@PathVariable Integer id) { return bookService.removeBookById(id); } }