2

我最近在 Django 中通过覆盖django.db.models.query.QuerySet和实现了软删除django.db.models.Manager

查询集:

class SoftDeleteQuerySet(QuerySet):
    def delete(self):
        super(SoftDeleteQuerySet, self).update(is_deleted=True, deleted=timezone.now())

    def delete_hard(self):
        super(SoftDeleteQuerySet, self).delete()

    def deleted(self):
        super(SoftDeleteQuerySet, self).filter(is_deleted=True)

经理:

class SoftDeletionManager(models.Manager):
    def __init__(self, *args, **kwargs):
        self.show_deleted = kwargs.pop('show_deleted', False)
        super(SoftDeletionManager, self).__init__(*args, **kwargs)

    def get_queryset(self):
        if self.show_deleted:
            return SoftDeleteQuerySet(self.model)
        return SoftDeleteQuerySet(self.model).filter(is_deleted=False)

    def delete_hard(self, instance):
        return self.get_queryset().delete_hard()

当我使用操作从 change_list 表单中删除任何对象时,这绝对可以正常工作。但是当我转到详细视图并尝试删除该对象时,它已完全从数据库中删除

到目前为止,我已经发现问题是由于MyModel.objects.get(...).delete()造成的。为了确保我没有走错方向,我在 python shell 中检查了它:

./manage.py 外壳:

>> Status.objects.all().count()
2
>> Status.all_objects.all().count()
4

>> Status.objects.get(pk=4).delete()
>> Status.objects.all().count()
1
>> Status.all_objects.all().count()
3

>> Status.objects.filter(pk=9).delete()
>> Status.objects.all().count()
0
>> Status.all_objects.all().count()
3

如果有人可以向我解释这种行为并建议我解决这个问题,那就太好了。

Django 1.6.5

Python 3.4.0

提前致谢 :)

4

1 回答 1

3

这个

Status.objects.get(pk=4).delete()

相当于这个

obj = Status.objects.get(pk=4) # obj is now Status object
obj.delete()

因此,在这种情况下,不会调用管理器方法。您delete直接调用未覆盖的对象。

然而

Status.objects.filter(pk=9).delete()

等于

qs = Status.objects.filter(pk=9) # qs is now SoftDeleteQuerySet object
qs.delete()

所以它确实调用了你的方法。

于 2014-07-18T18:24:50.287 回答