Django中的OneToOneField和ForeignKey有什么区别?
当前回答
学习新事物的最好和最有效的方法是看和研究现实世界的实际例子。假设您想在django中构建一个博客,记者可以在这里撰写和发布新闻文章。这家在线报纸的所有者希望允许他的每个记者发表尽可能多的文章,但不希望不同的记者为同一篇文章工作。这意味着当读者去阅读一篇文章时,他们只能在文章中看到一个作者。
例如:John的文章,Harry的文章,Rick的文章。你不能有Harry & Rick的文章,因为老板不希望两个或两个以上的作者在同一篇文章上工作。
如何在django的帮助下解决这个问题?解决这个问题的关键是django的ForeignKey。
下面是完整的代码,可以用来实现我们老板的想法。
from django.db import models
# Create your models here.
class Reporter(models.Model):
first_name = models.CharField(max_length=30)
def __unicode__(self):
return self.first_name
class Article(models.Model):
title = models.CharField(max_length=100)
reporter = models.ForeignKey(Reporter)
def __unicode__(self):
return self.title
运行python manage.py syncdb来执行sql代码,并在数据库中为你的应用程序构建表。然后使用python manage.py shell打开python shell。
创建Reporter对象R1。
In [49]: from thepub.models import Reporter, Article
In [50]: R1 = Reporter(first_name='Rick')
In [51]: R1.save()
创建Article对象A1。
In [5]: A1 = Article.objects.create(title='TDD In Django', reporter=R1)
In [6]: A1.save()
然后使用下面的代码段获取报告器的名称。
In [8]: A1.reporter.first_name
Out[8]: 'Rick'
现在通过运行以下python代码创建Reporter对象R2。
In [9]: R2 = Reporter.objects.create(first_name='Harry')
In [10]: R2.save()
现在尝试将R2添加到Article对象A1。
In [13]: A1.reporter.add(R2)
它不起作用,你会得到一个AttributeError,说'Reporter'对象没有属性'add'。
正如您所看到的,一个Article对象不能与多个Reporter对象相关。
R1呢?我们可以给它附加多个Article对象吗?
In [14]: A2 = Article.objects.create(title='Python News', reporter=R1)
In [15]: R1.article_set.all()
Out[15]: [<Article: Python News>, <Article: TDD In Django>]
这个实例向我们展示了如何使用django ForeignKey来定义多对一关系。
OneToOneField用于创建一对一的关系。
我们可以在上面的models.py文件中使用reporter = models.OneToOneField(reporter),但它在我们的例子中并不有用,因为作者将无法发布多篇文章。
每次要发布一篇新文章时,都必须创建一个新的Reporter对象。这很浪费时间,不是吗?
我强烈建议您尝试使用OneToOneField的示例,并了解其中的差异。我很确定在这个例子之后,你会完全知道django OneToOneField和django ForeignKey之间的区别。
其他回答
绘制项目之间关系的最简单方法是用简单的语言理解它们。例子
一个用户可以有很多辆车,但一辆车只能有一个车主。建立此关系后,应将外键用于关系较多的项上。在这种情况下,是汽车。这意味着您将在cars中包含user作为外键
一对一的关系很简单。说一个男人和一颗心。一个人只有一颗心,而一颗心只能属于一个人
OneToOneField:如果第二个表与
table2_col1 = models.OneToOneField(table1,on_delete=models.CASCADE, related_name='table1_id')
Table2将只包含一条对应于table1的pk值的记录,即table2_col1将有一个等于表pk的唯一值
table2_col1 == models.ForeignKey(table1, on_delete=models.CASCADE, related_name='table1_id')
Table2可以包含多条与table1的pk值相对应的记录。
OneToOneField(例如:一辆车有一个车主) ForeignKey(OneToMany)(例如:一个餐厅有很多项目)
此外,OneToOneField可用作主键以避免键重复。可能没有隐式/显式autofield
models.AutoField(primary_key=True)
但是使用OneToOneField作为主键(假设UserProfile模型为例):
user = models.OneToOneField(
User, null=False, primary_key=True, verbose_name='Member profile')
在Django最终指南中,OneToOneField(SomeModel)和ForeignKey(SomeModel, unique=True)的区别如下:
一对一字段 一对一的关系。从概念上讲,这类似于带有unique=True的ForeignKey,但是关系的“反向”端将直接返回单个对象。
与OneToOneField“反向”关系相反,ForeignKey“反向”关系返回一个QuerySet。
例子
例如,如果我们有以下两个模型(完整的模型代码如下):
汽车模型使用OneToOneField(引擎) Car2模型使用ForeignKey(Engine2, unique=True)
在python manage.py shell中执行以下命令:
一对一字段的例子
>>> from testapp.models import Car, Engine
>>> c = Car.objects.get(name='Audi')
>>> e = Engine.objects.get(name='Diesel')
>>> e.car
<Car: Audi>
ForeignKey with unique=True示例
>>> from testapp.models import Car2, Engine2
>>> c2 = Car2.objects.get(name='Mazda')
>>> e2 = Engine2.objects.get(name='Wankel')
>>> e2.car2_set.all()
[<Car2: Mazda>]
模型代码
from django.db import models
class Engine(models.Model):
name = models.CharField(max_length=25)
def __unicode__(self):
return self.name
class Car(models.Model):
name = models.CharField(max_length=25)
engine = models.OneToOneField(Engine)
def __unicode__(self):
return self.name
class Engine2(models.Model):
name = models.CharField(max_length=25)
def __unicode__(self):
return self.name
class Car2(models.Model):
name = models.CharField(max_length=25)
engine = models.ForeignKey(Engine2, unique=True, on_delete=models.CASCADE)
def __unicode__(self):
return self.name
推荐文章
- django MultiValueDictKeyError错误,我如何处理它
- 我如何在Django中创建一个鼻涕虫?
- 没有名为'django.core.urlresolvers'的模块
- Django - makemigrations -未检测到任何更改
- ">", "<", ">=" 和“< =”不工作”在Django过滤()”
- Django项目工作目录结构的最佳实践
- 如何在Django中表达一对多关系?
- 用Django创建电子邮件模板
- Django的嵌套元类是如何工作的?
- 用Django实现OpenID的最佳解决方案是什么?
- 在使用信号时,你不能在'原子'块的末尾执行查询,但只能在单元测试期间执行
- Django查询- id vs pk
- render(), render_to_response()和direct_to_template()之间有什么区别?
- 如何使用MySQLdb与Python和Django在OSX 10.6?
- 如何在Django中一次添加多个对象到ManyToMany关系?