34

我有一些简单的模型,Profile、Certifier 和 Designer,后者继承自 Profile(多表继承)。在 Designer 中,Certifier 有一个外键。

class Profile(models.Model):
    TYPES = (
        ('admin', _('Administrator')),
        ('certifier', _('Certifier')),
        ('designer', _('Designer'))
    )
    
    user = models.OneToOneField(User)
    type = models.CharField(max_length=9, choices=TYPES)
    
    def __str__(self):
        return self.user.username + ' (' + self.type + ')'

class Admin(Profile):
    pass
class Certifier(Profile):
    pass
class Designer(Profile):
    certifier = models.ForeignKey(Certifier)

在 Django 1.8 中,这非常有效,但在 1.9 中我得到了;

django.core.management.base.SystemCheckError: SystemCheckError: 系统检查发现了一些问题:

错误:

check.Designer.certifier:(models.E006)字段“certifier”与模型“check.profile”中的字段“certifier”冲突。

(在这种情况下,Profile.type 无关紧要,我只需要它来区分登录的用户配置文件类型)。

check.profile 显然没有“验证者”字段。这是一个错误还是我错过了什么?同样的事情发生在另一个项目中。

4

2 回答 2

28

我认为您不应该为该外键关系使用名称验证器,因为根据文档certifier,类 Profile 实际上具有,admin和字段(尽管通过描述符) ,在这种情况下,名称实际上会发生冲突。designer

from django.contrib.auth.models import User

c = Certifier.objects.create(
    type='admin',
    user=User.objects.latest('date_joined'),
)

p = c.profile_ptr
print(p.certifier) #username (admin)

更改为类似certifier_field = models.ForeignKey(Certifier)

正如评论中指出的那样,您可以将模型重命名为 CertifierProfile、AdminProfile 等以避免冲突。

或者您也可以通过添加到您的 来使支票静音,但这不是一个好方法。SILENCED_SYSTEM_CHECKS = ['models.E006']settings

于 2015-12-04T11:42:30.077 回答
23

你可以指定Profile是一个抽象类。这将防止支票与您的父字段混淆。

class Meta:
    abstract = True
于 2016-08-08T15:09:05.413 回答