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 处理批量软删除。
相关推荐
阿里嘎多哈基米7 小时前
SQL 层面行转列
数据库·sql·状态模式·mapper·行转列
抠脚学代码7 小时前
Ubuntu Qt x64平台搭建 arm64 编译套件
数据库·qt·ubuntu
jakeswang7 小时前
全解MySQL之死锁问题分析、事务隔离与锁机制的底层原理剖析
数据库·mysql
Heliotrope_Sun7 小时前
Redis
数据库·redis·缓存
一成码农8 小时前
MySQL问题7
数据库·mysql
吃饭最爱8 小时前
JUnit技术的核心和用法
数据库·oracle·sqlserver
专注API从业者8 小时前
Python/Java 代码示例:手把手教程调用 1688 API 获取商品详情实时数据
java·linux·数据库·python
java1234_小锋8 小时前
[免费]基于Python的协同过滤电影推荐系统(Django+Vue+sqlite+爬虫)【论文+源码+SQL脚本】
python·django·电影推荐系统·协同过滤
雨落Liy8 小时前
SQL 函数从入门到精通:原理、类型、窗口函数与实战指南
数据库·sql