亚洲免费在线视频-亚洲啊v-久久免费精品视频-国产精品va-看片地址-成人在线视频网

您的位置:首頁技術(shù)文章
文章詳情頁

基于Django OneToOneField和ForeignKey的區(qū)別詳解

瀏覽:3日期:2024-10-17 08:37:07

根據(jù)Django官方文檔介紹:

A one-to-one relationship. Conceptually, this is similar to a ForeignKey with unique=True, but the “reverse” side of the relation will directly return a single object.

OneToOneField與ForeignKey加上unique=True效果基本一樣,但是用OneToOneField反向關(guān)聯(lián)會直接返回對象。

相反地,使用ForeignKey, 反向關(guān)聯(lián)后會返回QuerySet。

例子:

from django.db import modelsclass Engine(models.Model): name = models.CharField(max_length=25) def __unicode__(self): return self.nameclass Car(models.Model): name = models.CharField(max_length=25) engine = models.OneToOneField(Engine) def __unicode__(self): return self.nameclass Engine2(models.Model): name = models.CharField(max_length=25) def __unicode__(self): return self.nameclass Car2(models.Model): name = models.CharField(max_length=25) engine = models.ForeignKey(Engine2, unique=True) def __unicode__(self): return self.name

在python manage.py shell里輸入:

>>> from testapp.models import Car, Engine>>> c = Car.objects.get(name=’Audi’)>>> e = Engine.objects.get(name=’Diesel’)>>> e.car # OneToOneField的反向關(guān)聯(lián)屬性如果沒有寫relate_name, 則是對方類名的小寫<Car: Audi>

>>> from testapp.models import Car2, Engine2>>> c2 = Car2.objects.get(name=’Mazda’)>>> e2 = Engine2.objects.get(name=’Wankel’)>>> e2.car2_set.all() # OneToOneField的反向關(guān)聯(lián)屬性如果沒有寫relate_name, 則是對方類名的小寫_set[<Car2: Mazda>]

補(bǔ)充知識:Django ForeignKey,ManyToManyField和OneToOneField的辨析

導(dǎo)引

模型(Models)是對網(wǎng)站所需信息種類的定義,其包含了網(wǎng)站存儲數(shù)據(jù)中的重要字段和數(shù)據(jù)行為。一般來說,一個模型對于數(shù)據(jù)庫中的一個表單。

字段(Fields)是模型的重要和唯一組成部分,他們由類別的屬性值所指定。

Field分類

由官方文檔Model field reference | Django Documentation定義:

Field一共分為AutoField、BinaryField、BooleanField、CharField、DateField、DecimalField、EmailField、FileField、FloatField、IntegerField、TextField、TimeField、URLField等類別,豐富的類別選項(xiàng)為數(shù)據(jù)庫存儲方式提供了完善的支持,而本文主要是針對如下三個關(guān)系型字段(Relationship fields):

關(guān)系型字段 對應(yīng)關(guān)系 ForeignKey 多對一 ManyToManyField 多對多 OneToOneField 一對一

分析

ForeignKey

首先查看源碼,在類的開頭有如下參數(shù):

many_to_many = Falsemany_to_one = Trueone_to_many = Falseone_to_one = False

由此可見,F(xiàn)oreignKey是many_to_one類型的,即“一對多”,我們引用官方文檔給出的示例:

from django.db import modelsclass Car(models.Model): manufacturer = models.ForeignKey( ’Manufacturer’, on_delete=models.CASCADE, ) # ...class Manufacturer(models.Model): # ... pass

由此我們可以看到,Car類型中有manufacturer字段,其類型是對應(yīng)Manufacturer類的ForeignKey。我們可以根據(jù)生活常識理解這種定義,由于一部汽車對應(yīng)一個生產(chǎn)商,而一個生產(chǎn)商可以對應(yīng)許多部汽車,所以兩者具有“一對多”的關(guān)系,在此種情況我們使用ForeignKey。

對于每個ForeignKey,我們需要給出關(guān)聯(lián)的模型和on_delete響應(yīng)的選項(xiàng),即

manufacturer = models.ForeignKey( ’Manufacturer’, on_delete=models.CASCADE, )

on_delete函數(shù)的作用是在此字段被刪除的時候做出的響應(yīng),其可選項(xiàng)如下:

選項(xiàng) 功能 CASCADE 級聯(lián)刪除,此類選項(xiàng)模仿SQL語句ON DELETE CASCADE,再刪除此字段信息的時候同時刪除包含F(xiàn)oreignKey字段的目標(biāo)(object) PROTECT 通過django.db.IntegrityError中的ProtectedError來保護(hù)此字段不被刪除,若進(jìn)行刪除操作則拋出錯誤 SET_NULL 將ForeignKey置為空,這只在null選項(xiàng)為True的時候產(chǎn)生作用 SET_DEFAULT 設(shè)為默認(rèn)值(default value),此默認(rèn)值已預(yù)先對ForeignKey設(shè)置 SET() 對ForeignKey設(shè)置對SET()函數(shù)傳遞的數(shù)值 DO_NOTHING 不進(jìn)行任何操作。若數(shù)據(jù)庫提高了引用完整性,則此種設(shè)置會拋出一個IntegrityError,除非對這一數(shù)據(jù)字段手動添加了SQL語句中的ON DELETE字段

還可以通過設(shè)置abstract屬性來定義一個抽象類:

from django.db import modelsclass AbstractCar(models.Model): manufacturer = models.ForeignKey(’Manufacturer’, on_delete=models.CASCADE) class Meta: abstract = True

ForeignKey還有如下的參數(shù)可以選擇:

參數(shù) 功能 limit_choices_to 通過一個限制對字段信息的某一可能選項(xiàng)進(jìn)行約束,可以通過字典,函數(shù)或者查詢值來設(shè)置 related_name 可以指定關(guān)聯(lián)的類在本類中的名稱,通過這一參數(shù)可以用兩個字段名引用同一個類,通過這個名稱父類可以取得子類的值,默認(rèn)為字段名 related_query_name 用于filter函數(shù)過濾和values函數(shù) to_field 關(guān)系關(guān)聯(lián)的相關(guān)對象名稱 db_constraint 控制在數(shù)據(jù)庫中是否應(yīng)該建立這一字段的約束 swappable 用于控制這一字段對于可交換類模型的行為

ManyToManyField

同樣在源碼中我們可以找到針對ManyToManyField的如下定義:

many_to_many = Truemany_to_one = Falseone_to_many = Falseone_to_one = False

由此可以知道,ManyToManyField是針對“many-to-many”即多對多關(guān)系定義的,它需要知道它關(guān)聯(lián)的類別。

官方文檔給出的示例代碼可以幫助理解:

from django.db import modelsclass Topping(models.Model): # ... passclass Pizza(models.Model): # ... toppings = models.ManyToManyField(Topping)

在示例代碼中,Pizza類的toppings字段由ManyToManyField與Toppings關(guān)聯(lián),我們可以由生活常識得出一片披薩上面會有很多種類的佐料,而一種佐料又可以用來制作多種披薩,兩者滿足“多對多”的關(guān)系。

ManyToManyField類有兩個經(jīng)常使用的參數(shù):through和through_fields,通過這兩個參數(shù)可以十分方便地建立中間項(xiàng)的關(guān)聯(lián),如示例代碼所示:

from django.db import modelsclass Person(models.Model): name = models.CharField(max_length=50)class Group(models.Model): name = models.CharField(max_length=128) members = models.ManyToManyField( Person, through=’Membership’, through_fields=(’group’, ’person’), )class Membership(models.Model): group = models.ForeignKey(Group, on_delete=models.CASCADE) person = models.ForeignKey(Person, on_delete=models.CASCADE) inviter = models.ForeignKey( Person, on_delete=models.CASCADE, related_name='membership_invites', ) invite_reason = models.CharField(max_length=64)

在Group類中有ManyToManyField類的字段members,這一字段通過through參數(shù)與membership聯(lián)系起來,后者表示“成員資格”,即表示“團(tuán)體”與“個人”之間關(guān)系的中間項(xiàng),而“through_fields”字段即為中間項(xiàng)連接起來的兩個類名,此處即group和person兩個類。

ManyToManyField還有以下參數(shù)可以選擇:

參數(shù) 功能 related_name 同F(xiàn)oreignKey,可以指定關(guān)聯(lián)的類在本類中的名稱 related_query_name 同F(xiàn)oreignKey,應(yīng)用于filter和values函數(shù) limit_choices_to 同F(xiàn)oreignKey,但如果自己定義了如“Membership”之類的中間類,則不會起到作用 symmetrical 對于迭代定義的ManyToManyField,其為這一字段建立一個單獨(dú)的屬性,而是設(shè)定symmetrical屬性為True,若期望使用此類迭代關(guān)系,可以手動設(shè)置其為False through 如上所示,用于設(shè)置中間項(xiàng)的名字,可以自己定義一個中間項(xiàng),若不定義的話系統(tǒng)也會分配一個中間項(xiàng) through_fields 通過元組來給出中間項(xiàng)關(guān)聯(lián)的兩個類名,可以查看上面的示例 db_table 可以通過這一屬性來手動設(shè)定保存這一字段的數(shù)據(jù)表名稱,若不設(shè)置則默認(rèn)為字段的名稱 db_contraint 是否在數(shù)據(jù)庫中建立約束 swappable 設(shè)置是否指向一個可交換的模型

OneToOneField

源碼中對OneToOneField的設(shè)置如下:

many_to_many = Falsemany_to_one = Falseone_to_many = Falseone_to_one = True

可知其是針對單對單的關(guān)系設(shè)定的字段。在概念上我們可以理解其為設(shè)置unique屬性為True的一種類型,區(qū)別之處在于它“反向”的數(shù)值會返回一個目標(biāo)值,這對于繼承關(guān)系的表達(dá)十分有用,例如一下示例程序:

from django.conf import settingsfrom django.db import modelsclass MySpecialUser(models.Model): user = models.OneToOneField( settings.AUTH_USER_MODEL, on_delete=models.CASCADE, ) supervisor = models.OneToOneField( settings.AUTH_USER_MODEL, on_delete=models.CASCADE, related_name=’supervisor_of’, )

OneToOneField既包含F(xiàn)oreignKey中的參數(shù),又包含一個額外的參數(shù)parent_link,若定義了一個類,其繼承了一個非抽象的類,而設(shè)置parent_link這個函數(shù)為True,則會將這個類視作繼承的類的父類,而不是一個新的OneToOneField。

以上這篇基于Django OneToOneField和ForeignKey的區(qū)別詳解就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持好吧啦網(wǎng)。

標(biāo)簽: Django
相關(guān)文章:
主站蜘蛛池模板: 最新99国产成人精品视频免费 | 亚洲aⅴ在线 | 男人扒开腿躁女人j | 男女视频免费看 | 日本三级11k影院在线 | 亚洲羞羞裸色私人影院 | 欧美天堂 | 国产日产韩产麻豆1区 | 亚洲一区二区三区四区在线观看 | 国产成人精品一区二区免费视频 | 欧美成人高清性色生活 | 可以免费看黄的网站 | 亚洲人成a在线网站 | 久香草视频在线观看免费 | 高清精品女厕在线观看 | 欧美成人国产一区二区 | 亚洲图片视频在线 | 国产一级特黄aaa大片 | аⅴ资源天堂8在线 | 日韩国产在线观看 | 国产视频网站在线观看 | 欧美成人影院 在线播放 | 久久国产免费观看精品3 | 2022国产精品网站在线播放 | 欧美一区二区三区高清视频 | 久久精品视频在线观看榴莲视频 | 欧美经典成人在观看线视频 | 国产亚洲欧美在线视频 | 日本特黄特色免费大片 | 成人 在线播放 | 欧美最爽乱淫视频播放黑人 | 国产日本韩国不卡在线视频 | 日本特黄特色高清免费视频 | 精品国产爱久久 | 国产亚洲欧美久久精品 | 久久国产精品视频 | 欧美一级高清毛片aaa | 一级色 | 欧美一区二区在线播放 | 九九精品视频一区在线 | 精品国产一区二区三区不卡 |