django 实现数据的逻辑删除

重写 queryset中的delete方法

from django.db import models
from django.db.models.query import QuerySet
from django.db.models.manager import BaseManager
class LyQuerySet(QuerySet):
    def delete(self):
        """
        重写QuerySet的 delete方法实现数据的逻辑删除
        QuerySet 自带的delete方法会返回两个值 
            1.删除数据的条数
            2.{app.model:删除的条数} 
            例如: 3, {home.Banner:3} 表示删除了home app下面的Banner表中3条数据
        我们可以不返回不会报错,如果要返回也可以模仿一下
        :return:
        """
        del_query = self._chain()
        del_query.update(is_del=True)

自己实现Manager方法

# 使用我们自己的QuerySet来管理各种操作 BaseManager.from_queryset(LyQuerySet)
class LyManager(BaseManager.from_queryset(LyQuerySet)):
    def __init__(self, *args, **kwargs):
        self.__add_is_del_filter = True
        super(LyManager, self).__init__(*args, **kwargs)

    def get_queryset(self, *args, **kwargs):

        queryset = super(LyManager, self).get_queryset(*args, **kwargs)

        # 判断用户是否主动传入is_del的筛选条件,如果主动传入,就不做自动筛选操作
        if self.__add_is_del_filter:
            # 过滤掉已删除的记录
            queryset = queryset.filter(is_del=False)

        return queryset

    def filter(self, *args, **kwargs):

        # 判断用户是否主动传入is_del的筛选条件,如果主动传入,就不做自动筛选操作
        if not kwargs.get('is_del') is None:
            self.__add_is_del_filter = False
        f = super(LyManager, self).filter(*args, **kwargs)
        return f

使用自己的Manager对象来管理models

class LyModel(models.Model):
    """
    继承这个类的Model可以实现
    is_del为True的数据过滤 和 数据的逻辑删除
    """
    create_time = models.DateTimeField(verbose_name='创建时间', auto_now_add=True)
    is_del = models.BooleanField(verbose_name='是否删除', default=False)
    # 使用对象.delete()时也实现逻辑删除
    def delete(self, using=None, keep_parents=False):
        self.is_del = True
        self.save()

    # 使用自己的Manager对象来管理models
    objects = LyManager()

    class Meta:
        abstract = True

 

在编写model时继承自己的Model就可以实现,默认过滤is_del为True的数据和数据的逻辑删除

class Banner(LyModel):
    title = models.CharField(max_length=500, verbose_name="广告标题", null=True, blank=True)
    link = models.CharField(max_length=500, verbose_name="广告链接", null=True, blank=True)
    image_url = models.ImageField(verbose_name="广告图片", upload_to=upload_path_handler(), null=True, blank=True)
    remark = models.TextField(verbose_name="备注信息", null=True, blank=True)
    is_show = models.BooleanField(default=False, verbose_name="是否显示")
    orders = models.IntegerField(default=1, verbose_name="排序")

    class Meta:
        db_table = "ly_banner"
        verbose_name = "轮播广告"
        verbose_name_plural = verbose_name

    def __str__(self):
        return self.title

 

posted on 2021-06-22 23:33  信奉上帝的小和尚  阅读(644)  评论(0编辑  收藏  举报

导航