欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

Django框架的關(guān)系型字段是什么-創(chuàng)新互聯(lián)

創(chuàng)新互聯(lián)www.cdcxhl.cn八線動態(tài)BGP香港云服務(wù)器提供商,新人活動買多久送多久,劃算不套路!

尚志ssl適用于網(wǎng)站、小程序/APP、API接口等需要進(jìn)行數(shù)據(jù)傳輸應(yīng)用場景,ssl證書未來市場廣闊!成為創(chuàng)新互聯(lián)的ssl證書銷售渠道,可以享受市場價(jià)格4-6折優(yōu)惠!如果有意向歡迎電話聯(lián)系或者加微信:13518219792(備注:SSL證書合作)期待與您的合作!

Django框架的關(guān)系型字段是什么?針對這個(gè)問題,這篇文章詳細(xì)介紹了相對應(yīng)的分析和解答,希望可以幫助更多想解決這個(gè)問題的小伙伴找到更簡單易行的方法。

除了我們前面說過的普通類型字段,Django還定義了一組關(guān)系類型字段,用來表示模型與模型之間的關(guān)系。

一、多對一(ForeignKey)

多對一的關(guān)系,通常被稱為外鍵。外鍵字段類的定義如下:

class ForeignKey(to, on_delete, **options)[source]

外鍵需要兩個(gè)位置參數(shù),一個(gè)是關(guān)聯(lián)的模型,另一個(gè)是on_delete選項(xiàng)。實(shí)際上,在目前版本中,on_delete選項(xiàng)也可以不設(shè)置,但Django極力反對如此,因此在Django2.0版本后,該選項(xiàng)會設(shè)置為必填。

外鍵要定義在‘多’的一方!

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

上面的例子中,每輛車都會有一個(gè)生產(chǎn)工廠,一個(gè)工廠可以生產(chǎn)N輛車,于是用一個(gè)外鍵字段manufacturer表示,并放在Car模型中。注意,此manufacturer非彼Manufacturer模型類,它是一個(gè)字段的名稱。在Django的模型定義中,經(jīng)常出現(xiàn)類似的英文單詞大小寫不同,一定要注意區(qū)分!

如果要關(guān)聯(lián)的對象在另外一個(gè)app中,可以顯式的指出。下例假設(shè)Manufacturer模型存在于production這個(gè)app中,則Car模型的定義如下:

class Car(models.Model):
    manufacturer = models.ForeignKey(        'production.Manufacturer',      # 關(guān)鍵在這里??!
        on_delete=models.CASCADE,
    )

如果要創(chuàng)建一個(gè)遞歸的外鍵,也就是自己關(guān)聯(lián)自己的的外鍵,使用下面的方法:

models.ForeignKey('self', on_delete=models.CASCADE)

核心在于‘self’這個(gè)引用。什么時(shí)候需要自己引用自己的外鍵呢?典型的例子就是評論系統(tǒng)!一條評論可以被很多人繼續(xù)評論,如下所示:

class Comment(models.Model):
    title = models.CharField(max_length=128)
    text = models.TextField()
    parent_comment = models.ForeignKey('self', on_delete=models.CASCADE)    # .....

注意上面的外鍵字段定義的是父評論,而不是子評論。為什么呢?因?yàn)橥怄I要放在‘多’的一方!

在實(shí)際的數(shù)據(jù)庫后臺,Django會為每一個(gè)外鍵添加_id后綴,并以此創(chuàng)建數(shù)據(jù)表里的一列。在上面的工廠與車的例子中,Car模型對應(yīng)的數(shù)據(jù)表中,會有一列叫做manufacturer_id。但實(shí)際上,在Django代碼中你不需要使用這個(gè)列名,除非你書寫原生的SQL語句,一般我們都直接使用字段名manufacturer。

關(guān)系字段的定義還有個(gè)小坑。在后面我們會講到的verbose_name參數(shù)用于設(shè)置字段的別名。很多情況下,為了方便,我們都會設(shè)置這么個(gè)值,并且作為字段的第一位置參數(shù)。但是對于關(guān)系字段,其第一位置參數(shù)永遠(yuǎn)是關(guān)系對象,不能是verbose_name,一定要注意!

參數(shù)說明:

外鍵還有一些重要的參數(shù),說明如下:

on_delete

當(dāng)一個(gè)被外鍵關(guān)聯(lián)的對象被刪除時(shí),Django將模仿on_delete參數(shù)定義的SQL約束執(zhí)行相應(yīng)操作。比如,你有一個(gè)可為空的外鍵,并且你想讓它在關(guān)聯(lián)的對象被刪除時(shí),自動設(shè)為null,可以如下定義:

user = models.ForeignKey(
    User,
    models.SET_NULL,
    blank=True,
    null=True,
)

該參數(shù)可選的值都內(nèi)置在django.db.models中,包括:

CASCADE:模擬SQL語言中的ON DELETE CASCADE約束,將定義有外鍵的模型對象同時(shí)刪除!(該操作為當(dāng)前Django版本的默認(rèn)操作?。?/p>

PROTECT:阻止上面的刪除操作,但是彈出ProtectedError異常。

SET_NULL:將外鍵字段設(shè)為null,只有當(dāng)字段設(shè)置了null=True時(shí),方可使用該值。

SET_DEFAULT:將外鍵字段設(shè)為默認(rèn)值。只有當(dāng)字段設(shè)置了default參數(shù)時(shí),方可使用。

DO_NOTHING:什么也不做。

SET():設(shè)置為一個(gè)傳遞給SET()的值或者一個(gè)回調(diào)函數(shù)的返回值。注意大小寫。

from django.conf import settings
from django.contrib.auth import get_user_model
from django.db import models
def get_sentinel_user():
    return get_user_model().objects.get_or_create(username='deleted')[0]
    class MyModel(models.Model):
    user = models.ForeignKey(
        settings.AUTH_USER_MODEL,
        on_delete=models.SET(get_sentinel_user),
    )

limit_choices_to

該參數(shù)用于限制外鍵所能關(guān)聯(lián)的對象,只能用于Django的ModelForm(Django的表單模塊)和admin后臺,對其它場合無限制功能。其值可以是一個(gè)字典、Q對象或者一個(gè)返回字典或Q對象的函數(shù)調(diào)用,如下例所示:

staff_member = models.ForeignKey(
    User,
    on_delete=models.CASCADE,
    limit_choices_to={'is_staff': True},
)

這樣定義,則ModelForm的staff_member字段列表中,只會出現(xiàn)那些is_staff=True的Users對象,這一功能對于admin后臺非常有用。

可以參考下面的方式,使用函數(shù)調(diào)用:

def limit_pub_date_choices():
    return {'pub_date__lte': datetime.date.utcnow()}
    # ...limit_choices_to = limit_pub_date_choices# ...

related_name

用于關(guān)聯(lián)對象反向引用模型的名稱。以前面車和工廠的例子解釋,就是從工廠反向關(guān)聯(lián)到車的關(guān)系名稱。

通常情況下,這個(gè)參數(shù)我們可以不設(shè)置,Django會默認(rèn)以模型的小寫作為反向關(guān)聯(lián)名,比如對于工廠就是car,如果你覺得car還不夠直觀,可以如下定義:

class Car(models.Model):
    manufacturer = models.ForeignKey(        'production.Manufacturer',      
        on_delete=models.CASCADE,
        related_name='car_producted_by_this_manufacturer',  # 看這里??!
    )

也許我定義了一個(gè)蹩腳的詞,但表達(dá)的意思很清楚。以后從工廠對象反向關(guān)聯(lián)到它所生產(chǎn)的汽車,就可以使用maufacturer.car_producted_by_this_manufacturer了。

如果你不想為外鍵設(shè)置一個(gè)反向關(guān)聯(lián)名稱,可以將這個(gè)參數(shù)設(shè)置為“+”或者以“+”結(jié)尾,如下所示:

user = models.ForeignKey(
    User,
    on_delete=models.CASCADE,
    related_name='+',
)

related_query_name

反向關(guān)聯(lián)查詢名。用于從目標(biāo)模型反向過濾模型對象的名稱。(過濾和查詢在后續(xù)章節(jié)會介紹)

class Tag(models.Model):
    article = models.ForeignKey(
        Article,
        on_delete=models.CASCADE,
        related_name="tags",
        related_query_name="tag",       # 注意這一行
    )
    name = models.CharField(max_length=255)# 現(xiàn)在可以使用‘tag’作為查詢名了
    Article.objects.filter(tag__name="important")

to_field

默認(rèn)情況下,外鍵都是關(guān)聯(lián)到被關(guān)聯(lián)對象的主鍵上(一般為id)。如果指定這個(gè)參數(shù),可以關(guān)聯(lián)到指定的字段上,但是該字段必須具有unique=True屬性,也就是具有唯一屬性。

db_constraint

默認(rèn)情況下,這個(gè)參數(shù)被設(shè)為True,表示遵循數(shù)據(jù)庫約束,這也是大多數(shù)情況下你的選擇。如果設(shè)為False,那么將無法保證數(shù)據(jù)的完整性和合法性。在下面的場景中,你可能需要將它設(shè)置為False:

有歷史遺留的不合法數(shù)據(jù),沒辦法的選擇你正在分割數(shù)據(jù)表

當(dāng)它為False,并且你試圖訪問一個(gè)不存在的關(guān)系對象時(shí),會拋出DoesNotExist 異常。

swappable

控制遷移框架的動作,如果當(dāng)前外鍵指向一個(gè)可交換的模型。使用場景非常稀少,通常請將該參數(shù)保持默認(rèn)的True。

二、多對多(ManyToManyField)

多對多關(guān)系在數(shù)據(jù)庫中也是非常常見的關(guān)系類型。比如一本書可以有好幾個(gè)作者,一個(gè)作者也可以寫好幾本書。多對多的字段可以定義在任何的一方,請盡量定義在符合人們思維習(xí)慣的一方,但不要同時(shí)都定義。

class ManyToManyField(to, **options)[source]

多對多關(guān)系需要一個(gè)位置參數(shù):關(guān)聯(lián)的對象模型。它的用法和外鍵多對一基本類似。

在數(shù)據(jù)庫后臺,Django實(shí)際上會額外創(chuàng)建一張用于體現(xiàn)多對多關(guān)系的中間表。默認(rèn)情況下,該表的名稱是“多對多字段名+關(guān)聯(lián)對象模型名+一個(gè)獨(dú)一無二的哈希碼”,例如‘a(chǎn)uthor_books_9cdf4’,當(dāng)然你也可以通過db_table選項(xiàng),自定義表名。

參數(shù)說明:

related_name

參考外鍵的相同參數(shù)。

related_query_name

參考外鍵的相同參數(shù)。

limit_choices_to

參考外鍵的相同參數(shù)。但是對于使用through參數(shù)自定義中間表的多對多字段無效。

symmetrical

默認(rèn)情況下,Django中的多對多關(guān)系是對稱的。看下面的例子:

from django.db import modelsclass Person(models.Model):
    friends = models.ManyToManyField("self")

Django認(rèn)為,如果我是你的朋友,那么你也是我的朋友,這是一種對稱關(guān)系,Django不會為Person模型添加person_set屬性用于反向關(guān)聯(lián)。如果你不想使用這種對稱關(guān)系,可以將symmetrical設(shè)置為False,這將強(qiáng)制Django為反向關(guān)聯(lián)添加描述符。

through

(定義中間表)

如果你想自定義多對多關(guān)系的那張額外的關(guān)聯(lián)表,可以使用這個(gè)參數(shù)!參數(shù)的值為一個(gè)中間模型。

最常見的使用場景是你需要為多對多關(guān)系添加額外的數(shù)據(jù),比如兩個(gè)人建立QQ好友的時(shí)間。

通常情況下,這張表在數(shù)據(jù)庫內(nèi)的結(jié)構(gòu)是這個(gè)樣子的:

中間表的id列....模型對象的id列.....被關(guān)聯(lián)對象的id列# 各行數(shù)據(jù)

如果自定義中間表并添加時(shí)間字段,則在數(shù)據(jù)庫內(nèi)的表結(jié)構(gòu)如下:

中間表的id列....模型對象的id列.....被關(guān)聯(lián)對象的id列.....時(shí)間對象列# 各行數(shù)據(jù)

看下面的例子:

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)

上面的代碼中,通過class Membership(models.Model)定義了一個(gè)新的模型,用來保存Person和Group模型的多對多關(guān)系,并且同時(shí)增加了‘邀請人’和‘邀請時(shí)間’的字段。

through參數(shù)在某些使用場景中是必須的,至關(guān)重要,請務(wù)必掌握!

through_fields

接著上面的例子。Membership模型中包含兩個(gè)關(guān)聯(lián)Person的外鍵,Django無法確定到底使用哪個(gè)作為和Group關(guān)聯(lián)的對象。所以,在這個(gè)例子中,必須顯式的指定through_fields參數(shù),用于定義關(guān)系。

through_fields參數(shù)接收一個(gè)二元元組('field1', 'field2'),field1是指向定義有多對多關(guān)系的模型的外鍵字段的名稱,這里是Membership中的‘group’字段(注意大小寫),另外一個(gè)則是指向目標(biāo)模型的外鍵字段的名稱,這里是Membership中的‘person’,而不是‘inviter’。

再通俗的說,就是through_fields參數(shù)指定從中間表模型Membership中選擇哪兩個(gè)字段,作為關(guān)系連接字段。

db_table

設(shè)置中間表的名稱。不指定的話,則使用默認(rèn)值。

db_constraint

參考外鍵的相同參數(shù)。

swappable

參考外鍵的相同參數(shù)。

ManyToManyField多對多字段不支持Django內(nèi)置的validators驗(yàn)證功能。

null參數(shù)對ManyToManyField多對多字段無效!設(shè)置null=True毫無意義

三、一對一(OneToOneField)

一對一關(guān)系類型的定義如下:

class OneToOneField(to, on_delete, parent_link=False, **options)[source]

從概念上講,一對一關(guān)系非常類似具有unique=True屬性的外鍵關(guān)系,但是反向關(guān)聯(lián)對象只有一個(gè)。這種關(guān)系類型多數(shù)用于當(dāng)一個(gè)模型需要從別的模型擴(kuò)展而來的情況。比如,Django自帶auth模塊的User用戶表,如果你想在自己的項(xiàng)目里創(chuàng)建用戶模型,又想方便的使用Django的認(rèn)證功能,那么一個(gè)比較好的方案就是在你的用戶模型里,使用一對一關(guān)系,添加一個(gè)與auth模塊User模型的關(guān)聯(lián)字段。

該關(guān)系的第一位置參數(shù)為關(guān)聯(lián)的模型,其用法和前面的多對一外鍵一樣。

如果你沒有給一對一關(guān)系設(shè)置related_name參數(shù),Django將使用當(dāng)前模型的小寫名作為默認(rèn)值。

看下面的例子:

from django.conf import settings
from django.db import models# 兩個(gè)字段都使用一對一關(guān)聯(lián)到了Django內(nèi)置的auth模塊中的User模型
class 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',
    )

這樣下來,你的User模型將擁有下面的屬性:

>>> user = User.objects.get(pk=1)
>>> hasattr(user, 'myspecialuser')
True
>>> hasattr(user, 'supervisor_of')
True

OneToOneField一對一關(guān)系擁有和多對一外鍵關(guān)系一樣的額外可選參數(shù),只是多了一個(gè)parent_link參數(shù)。

跨模塊的模型:

有時(shí)候,我們關(guān)聯(lián)的模型并不在當(dāng)前模型的文件內(nèi),沒關(guān)系,就像我們導(dǎo)入第三方庫一樣的從別的模塊內(nèi)導(dǎo)入進(jìn)來就好,如下例所示:

from django.db import models
from geography.models import ZipCodeclass Restaurant(models.Model):
    # ...
    zip_code = models.ForeignKey(        
    ZipCode,        
    on_delete=models.SET_NULL,        
    blank=True,        
    null=True,
    )

關(guān)于Django框架的關(guān)系型字段是什么問題的解答就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關(guān)注創(chuàng)新互聯(lián)-成都網(wǎng)站建設(shè)公司行業(yè)資訊頻道了解更多相關(guān)知識。

文章名稱:Django框架的關(guān)系型字段是什么-創(chuàng)新互聯(lián)
網(wǎng)頁網(wǎng)址:http://aaarwkj.com/article0/ccheio.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)頁設(shè)計(jì)公司、網(wǎng)站導(dǎo)航網(wǎng)站維護(hù)、云服務(wù)器、網(wǎng)站制作、網(wǎng)站收錄

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

成都網(wǎng)頁設(shè)計(jì)公司
日韩欧美精品视频一区| 日本成熟妇高潮视频在线观看不卡 | 欧美日韩黄色的三级视频| 人妻黄色这里只有精品| 精品国产50部农村老熟女av| 午夜福利精品在线观看| 国产乱码精品一区二区蜜臀| 亚洲国产男同日韩小鲜肉| 少妇特黄a一区二区三区| 日韩精品第一区第二区| 中文岳妇荡欲丰满肥熟| 日韩不卡高清免费在线视频| 日韩蜜桃av一二三四区| 中文字幕日韩欧美一区| 日本东京热不卡一区二区| 国产一区在线免费在线观看| 中文字幕五月婷婷免费| 久草尤物视频在线观看| 天天躁人人躁夜夜躁狠狠躁| 欧洲一区二区在线激情| 青青草青青草在线观看视频| 成熟人妻一区二区三区人妻| 国产午夜精品自拍视频| 亚洲一区二区日韩人妻| 欧美高清一区二区三区不卡| 欧美精品久久久久九九九| 国产极白丝白浆日本国产| 中国亚洲黄色录像免费看| 韩国三级伦理中文字幕| 人妻少妇一区二区三区四区| 日韩视频一区二区三区系列| 97精品少妇一区二区三区| 国产免费播放一区二区三区| 黄色大片黄色大片黄色大片| 欧美一区二区国产精品日韩| 懂色一区二区三区精品视频| 成年人黄色免费网站在线观看| 日本精品在线一区二区| 日韩精品中文字幕人妻系列| 亚洲三级黄片在线观看| 国产日韩在线不卡网站|