OneToOneField和删除

Nic*_*ick 36 django

我有以下型号:

from django.db import models
from django.contrib.auth.models import User


class Profile(models.Model):
    user = models.OneToOneField(User)
    # ...

    def __unicode__(self):
        return u'%s %s' % (self.user.first_name, self.user.last_name)
Run Code Online (Sandbox Code Playgroud)

当使用Django管理员删除用户时,配置文件也会被删除,这就是我想要的.但是,使用Django管理删除配置文件时,用户也不会被删除,这是不是我想要的.如何删除配置文件也会删除用户?

mik*_*725 53

由于Profile链接User,它是关系中的依赖模型.因此,删除用户时,将删除所有相关模型.但是,当您删除配置文件时,由于User不依赖于配置文件,因此不会删除它.

不幸的是,根据on_deleteDjango的文档,没有on_delete删除父母关系的规则.为了做到这一点,你可以覆盖Profiledelete方法:

class Profile(models.Model):
    # ...

    def delete(self, *args, **kwargs):
        self.user.delete()
        return super(self.__class__, self).delete(*args, **kwargs)
Run Code Online (Sandbox Code Playgroud)

然后做:

Profile.objects.get(...).delete()
Run Code Online (Sandbox Code Playgroud)

还将删除配置文件的用户.但是,在delete使用查询集(这是Django Admin中调用的)删除配置文件时,不会调用该方法,因为Django使用SQL DELETE批量删除对象:

Profile.objects.filter(...).delete()
Run Code Online (Sandbox Code Playgroud)

在这种情况下,根据Django docs的建议,您将不得不使用post_delete信号(docs).

from django.dispatch import receiver
from django.db.models.signals import post_delete

@receiver(post_delete, sender=Profile)
def post_delete_user(sender, instance, *args, **kwargs):
    if instance.user: # just in case user is not specified
        instance.user.delete()
Run Code Online (Sandbox Code Playgroud)


spe*_*son 10

使用Profile删除方法上的信号去删除相关用户:

from django.db.models.signals import post_delete

def delete_related_user(sender, **kwargs):
    deleted_profile = kwargs['instance']
    deleted_profile.user.delete()

post_delete.connect(delete_related_user, sender=Profile)
Run Code Online (Sandbox Code Playgroud)