Django中的软删除

软删除(Soft Delete)是一种数据删除策略,它并不真正从数据库中删除记录,而是通过标记(如 is_deleted 字段)来表示记录已被删除。

这样做的好处是可以保留数据历史,支持数据恢复和审计。

在 Django 里可以通过 自定义 Manager + 重写 delete 方法 来实现。


1. 在模型里增加 is_delete 字段

python 复制代码
from django.db import models

class BaseModel(models.Model):
    is_delete = models.BooleanField(default=False, verbose_name="是否删除")

    class Meta:
        abstract = True  # 抽象基类,不会建表

这样所有继承 BaseModel 的表都有 is_delete 字段。


2. 自定义 Manager(默认过滤掉删除的记录)

python 复制代码
class ActiveManager(models.Manager):
    def get_queryset(self):
        # 默认只返回 is_delete=False 的数据
        return super().get_queryset().filter(is_delete=False)

3. 在模型里应用 Manager

python 复制代码
class User(BaseModel):
    name = models.CharField(max_length=100)

    # managers
    objects = ActiveManager()   # 默认只取未删除的
    all_objects = models.Manager()  # 需要时可以取所有(包括已删除的)

这样:

python 复制代码
User.objects.all()        # 只会查 is_delete=False
User.all_objects.all()    # 不加过滤,所有数据都能查

4. 重写 delete() 方法(软删除)

python 复制代码
class User(BaseModel):
    name = models.CharField(max_length=100)

    objects = ActiveManager()
    all_objects = models.Manager()

    def delete(self, using=None, keep_parents=False):
        self.is_delete = True
        self.save(update_fields=['is_delete'])  # 只更新 is_delete 字段

这样:

python 复制代码
u = User.objects.get(id=1)
u.delete()  # 不会真正删除,只会把 is_delete 置为 True

5. 如果要做批量删除

Django 的 QuerySet.delete() 默认会直接删掉数据,所以我们也可以自定义一个 QuerySet 来支持批量软删除:

python 复制代码
class SoftDeleteQuerySet(models.QuerySet):
    def delete(self):
        return super().update(is_delete=True)

结合 Manager 使用:

python 复制代码
class ActiveManager(models.Manager):
    def get_queryset(self):
        return SoftDeleteQuerySet(self.model, using=self._db).filter(is_delete=False)

这样就支持:

python 复制代码
User.objects.filter(name="Tom").delete()  # 也会变成软删除

总结

  1. is_delete 字段标记是否删除。
  2. 自定义 Manager 过滤掉已删除的数据。
  3. 重写 delete() 实现软删除逻辑。
  4. 如有需要,配合自定义 QuerySet 处理批量软删除。
相关推荐
自不量力的A同学30 分钟前
Redisson 4.2.0 发布,官方推荐的 Redis 客户端
数据库·redis·缓存
Exquisite.32 分钟前
Mysql
数据库·mysql
全栈前端老曹1 小时前
【MongoDB】深入研究副本集与高可用性——Replica Set 架构、故障转移、读写分离
前端·javascript·数据库·mongodb·架构·nosql·副本集
R1nG8631 小时前
CANN资源泄漏检测工具源码深度解读 实战设备内存泄漏排查
数据库·算法·cann
阿钱真强道1 小时前
12 JetLinks MQTT直连设备事件上报实战(继电器场景)
linux·服务器·网络·数据库·网络协议
逍遥德2 小时前
Sring事务详解之02.如何使用编程式事务?
java·服务器·数据库·后端·sql·spring
笨蛋不要掉眼泪2 小时前
Redis哨兵机制全解析:原理、配置与实战故障转移演示
java·数据库·redis·缓存·bootstrap
Coder_Boy_2 小时前
基于SpringAI的在线考试系统-整体架构优化设计方案
java·数据库·人工智能·spring boot·架构·ddd
fen_fen10 小时前
Oracle建表语句示例
数据库·oracle
砚边数影12 小时前
数据可视化入门:Matplotlib 基础语法与折线图绘制
数据库·信息可视化·matplotlib·数据可视化·kingbase·数据库平替用金仓·金仓数据库