数据库防止误删操作——打数据标记

数据库防止误删操作——打数据标记

本文讲解,如何在数据库层面上通过,打数据标记的方法,防止用户的数据误删的情况。

简介

对于数据库删除操作,在很多情况下并不是真正的删除,而是将数据标记为删除状态,以防止误删或者恢复数据。这样做的好处是可以节省物理存储空间,并且避免了数据被永久性删除带来的风险。在实现上也比直接删除更为方便,同时通过备份策略保留历史增量备份,则也可以完整记录历史变化。

逻辑删除:逻辑删除即将数据的删除标记位设置为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);
    }
}

猜你喜欢

转载自blog.csdn.net/qq_51447496/article/details/131058180
今日推荐