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 处理批量软删除。
相关推荐
清风6666663 小时前
基于单片机的双机串口通信与数字串存储系统设计
数据库·单片机·mongodb·毕业设计·课程设计·期末大作业
数据库知识分享者小北3 小时前
AI Agent越用越笨?阿里云AnalyticDB「AI上下文工程」一招破解!
数据库
一匹电信狗4 小时前
【MySQL】数据库表的操作
linux·运维·服务器·数据库·mysql·ubuntu·小程序
api_180079054604 小时前
性能优化揭秘:将淘宝商品 API 响应时间从 500ms 优化到 50ms 的技术实践
大数据·数据库·性能优化·数据挖掘
白衣鸽子4 小时前
MySQL 时间类型深度解析:精度、时区陷阱与版本兼容
数据库·后端·mysql
冲上云霄的Jayden4 小时前
MySQL InnoDB 状态(SHOW ENGINE INNODB STATUS)深度分析与性能优化建议
数据库·mysql·性能优化·innodb
元闰子5 小时前
怎么让程序更高效地连起来?
数据库·redis·mysql
洲覆5 小时前
Redis 内存淘汰策略
开发语言·数据库·redis·缓存
胖头鱼的鱼缸(尹海文)6 小时前
数据库管理-第376期 Oracle AI DB 23.26新特性一览(20251016)
数据库·人工智能·oracle
麦聪聊数据6 小时前
浅谈SQL审核(一):SQL审核实现方式与常见工具的选择
数据库·sql