【问题标题】:OneToOneField and DeletingOneToOneField 和删除
【发布时间】:2026-01-31 02:05:04
【问题描述】:

我有以下型号:

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)

当使用 Django 管理员删除用户时,配置文件也会被删除,这正是我想要的。但是,当使用 Django 管理员删除配置文件时,用户确实 not 被删除,这 not 是我想要的。我怎样才能做到删除配置文件也会删除用户?

【问题讨论】:

    标签: django


    【解决方案1】:

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

    不幸的是,根据on_delete Django docs,没有删除父关系的on_delete 规则。为此,您可以覆盖Profiledelete 方法:

    class Profile(models.Model):
        # ...
    
        def delete(self, *args, **kwargs):
            self.user.delete()
            return super(self.__class__, self).delete(*args, **kwargs)
    

    那么当做:

    Profile.objects.get(...).delete()
    

    还将删除配置文件的用户。然而,delete 方法不会在使用查询集(在 Django Admin 中调用)删除配置文件时被调用,因为那时 Django 使用 SQL DELETE 批量删除对象:

    Profile.objects.filter(...).delete()
    

    在这种情况下,按照 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()
    

    【讨论】:

    • 非常感谢!我还是 Django 的新手,所以如果任何其他 Django 新手发现这个,我必须在 @receiver 装饰器中添加一个 dispatch_uid 才能让它工作。但在那之后,效果很好!
    • @Nick 你能用dispatch_uid详细说明你的解决方案吗,因为删除大量用户它不起作用
    • 在更改列表中使用下拉“删除”操作时会调用信号吗?是的!
    【解决方案2】:

    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)
    

    【讨论】: