我现在正在定义我的Django模型,我意识到在模型字段类型中没有OneToManyField。我相信有办法做到这一点,所以我不确定我错过了什么。我基本上有这样的东西:
class Dude(models.Model):
numbers = models.OneToManyField('PhoneNumber')
class PhoneNumber(models.Model):
number = models.CharField()
在这种情况下,每个Dude可以有多个PhoneNumber,但这种关系应该是单向的,因为我不需要从PhoneNumber中知道哪个Dude拥有它,本质上,因为我可能有许多不同的对象拥有PhoneNumber实例,例如Business:
class Business(models.Model):
numbers = models.OneToManyField('PhoneNumber')
我将在模型中用什么替换OneToManyField(它不存在)来表示这种关系?我来自Hibernate/JPA,在那里声明一对多关系非常简单:
@OneToMany
private List<PhoneNumber> phoneNumbers;
如何在Django中表达这一点?
要在Django中处理一对多关系,你需要使用ForeignKey。
关于ForeignKey的文档是非常全面的,应该回答你所有的问题:
https://docs.djangoproject.com/en/3.2/ref/models/fields/#foreignkey
示例中的当前结构允许每个Dude拥有一个数字,并且每个数字属于多个Dude(与Business相同)。
如果你想要相反的关系,你需要给PhoneNumber模型添加两个ForeignKey字段,一个给Dude,一个给Business。这将允许每个号码属于一个人或一个企业,并使人和企业能够拥有多个号码。我觉得这可能就是你要找的东西。
class Business(models.Model):
...
class Dude(models.Model):
...
class PhoneNumber(models.Model):
dude = models.ForeignKey(Dude)
business = models.ForeignKey(Business)
在Django中,一对多关系被称为ForeignKey。但是,它只在一个方向上工作,因此您需要的不是类Dude的number属性
class Dude(models.Model):
...
class PhoneNumber(models.Model):
dude = models.ForeignKey(Dude)
许多模型都可以有一个指向另一个模型的ForeignKey,所以有第二个PhoneNumber属性是有效的,这样
class Business(models.Model):
...
class Dude(models.Model):
...
class PhoneNumber(models.Model):
dude = models.ForeignKey(Dude)
business = models.ForeignKey(Business)
您可以使用d.f onenumber_set.objects.all()访问Dude对象d的PhoneNumbers,然后对业务对象执行类似的操作。
虽然《滚石》杂志的回答很好,直截了当,而且很实用,但我认为有两件事它没有解决。
如果OP想要强制一个电话号码不能同时属于Dude和Business
由于在PhoneNumber模型上定义关系而不是在Dude/Business模型上定义关系而产生的不可避免的悲伤感觉。当外星人来到地球时,我们想要添加一个Alien模型,我们需要修改PhoneNumber(假设外星人有电话号码),而不是简单地向Alien模型添加一个“phone_numbers”字段。
介绍内容类型框架,它公开了一些对象,允许我们在PhoneNumber模型上创建“通用外键”。然后,我们可以在Dude和Business上定义反向关系
from django.contrib.contenttypes.fields import GenericForeignKey, GenericRelation
from django.contrib.contenttypes.models import ContentType
from django.db import models
class PhoneNumber(models.Model):
number = models.CharField()
content_type = models.ForeignKey(ContentType, on_delete=models.CASCADE)
object_id = models.PositiveIntegerField()
owner = GenericForeignKey()
class Dude(models.Model):
numbers = GenericRelation(PhoneNumber)
class Business(models.Model):
numbers = GenericRelation(PhoneNumber)
有关详细信息,请参阅文档,也可以查看这篇文章以获得快速教程。
此外,这里有一篇文章反对使用通用fk。
首先我们来参观一下:
01)一对多关系:
ASSUME:
class Business(models.Model):
name = models.CharField(max_length=200)
.........
.........
phone_number = models.OneToMany(PhoneNumber) (NB: Django do not support OneToMany relationship)
class Dude(models.Model):
name = models.CharField(max_length=200)
.........
.........
phone_number = models.OneToMany(PhoneNumber) (NB: Django do not support OneToMany relationship)
class PhoneNumber(models.Model):
number = models.CharField(max_length=20)
........
........
注意:Django没有提供任何OneToMany关系。所以在Django中不能使用upper方法。但是我们需要在关系模型中进行转换。那么我们能做什么呢?在这种情况下,我们需要将关系模型转换为反向关系模型。
在这里:
关系模型=一个多
所以,反向关系模型=多对一
注意:Django支持多对一关系&在Django中多对一由ForeignKey表示。
02)多对一关系:
SOLVE:
class Business(models.Model):
.........
.........
class Dude(models.Model):
.........
.........
class PhoneNumber(models.Model):
........
........
business = models.ForeignKey(Business)
dude = models.ForeignKey(Dude)
Nb:简单想想!!
尽管这里的大多数答案在定义关系时传递了允许您定义OneToManyRelationship的关系,但它们实际上是多对多关系的定义。
下面是一个更准确的实现:
class Dude(models.Model):
name = models.CharField(max_length=256, unique=True)
class PhoneNumber(models.Model):
number = models.CharField(max_length=256, unique=True)
class DudePhoneNumbers(models.Model):
dude = models.ForeignKey(Dude, related_name="phone_number_relations", on_delete=models.CASCADE)
phone_number = models.OneToOneField(PhoneNumber, related_name="dude_relation", on_delete=models.CASCADE)