外鍵和表關(guān)系
外鍵:
在MySQL
中,表有兩種引擎漾肮,一種是InnoDB
厂抖,另外一種是myisam
。如果使用的是InnoDB
引擎克懊,是支持外鍵約束的忱辅。外鍵的存在使得ORM
框架在處理表關(guān)系的時(shí)候異常的強(qiáng)大七蜘。因此這里我們首先來(lái)介紹下外鍵在Django
中的使用。
類(lèi)定義為class ForeignKey(to,on_delete,**options)
墙懂。第一個(gè)參數(shù)是引用的是哪個(gè)模型橡卤,第二個(gè)參數(shù)是在使用外鍵引用的模型數(shù)據(jù)被刪除了,這個(gè)字段該如何處理损搬,比如有CASCADE
碧库、SET_NULL
等。這里以一個(gè)實(shí)際案例來(lái)說(shuō)明巧勤。比如有一個(gè)User
和一個(gè)Article
兩個(gè)模型嵌灰。一個(gè)User
可以發(fā)表多篇文章,一個(gè)Article
只能有一個(gè)Author
颅悉,并且通過(guò)外鍵進(jìn)行引用伞鲫。那么相關(guān)的示例代碼如下:
class User(models.Model):
username = models.CharField(max_length=20)
password = models.CharField(max_length=100)
class Article(models.Model):
title = models.CharField(max_length=100)
content = models.TextField()
author = models.ForeignKey("User",on_delete=models.CASCADE)
以上使用ForeignKey
來(lái)定義模型之間的關(guān)系。即在article
的實(shí)例中可以通過(guò)author
屬性來(lái)操作對(duì)應(yīng)的User
模型签舞。這樣使用起來(lái)非常的方便。示例代碼如下:
article = Article(title='abc',content='123')
author = User(username='張三',password='111111')
article.author = author
article.save()
# 修改article.author上的值
article.author.username = '李四'
article.save()
為什么使用了ForeignKey
后柒瓣,就能通過(guò)author
訪問(wèn)到對(duì)應(yīng)的user
對(duì)象呢儒搭。因此在底層,Django
為Article
表添加了一個(gè)屬性名_id
的字段(比如author的字段名稱(chēng)是author_id)芙贫,這個(gè)字段是一個(gè)外鍵搂鲫,記錄著對(duì)應(yīng)的作者的主鍵。以后通過(guò)article.author
訪問(wèn)的時(shí)候磺平,實(shí)際上是先通過(guò)author_id
找到對(duì)應(yīng)的數(shù)據(jù)魂仍,然后再提取User
表中的這條數(shù)據(jù),形成一個(gè)模型拣挪。
如果想要引用另外一個(gè)app
的模型擦酌,那么應(yīng)該在傳遞to
參數(shù)的時(shí)候,使用app.model_name
進(jìn)行指定菠劝。以上例為例赊舶,如果User
和Article
不是在同一個(gè)app
中,那么在引用的時(shí)候的示例代碼如下:
# User模型在user這個(gè)app中
class User(models.Model):
username = models.CharField(max_length=20)
password = models.CharField(max_length=100)
# Article模型在article這個(gè)app中
class Article(models.Model):
title = models.CharField(max_length=100)
content = models.TextField()
author = models.ForeignKey("user.User",on_delete=models.CASCADE)
如果模型的外鍵引用的是本身自己這個(gè)模型赶诊,那么to
參數(shù)可以為'self'
笼平,或者是這個(gè)模型的名字。在論壇開(kāi)發(fā)中舔痪,一般評(píng)論都可以進(jìn)行二級(jí)評(píng)論寓调,即可以針對(duì)另外一個(gè)評(píng)論進(jìn)行評(píng)論,那么在定義模型的時(shí)候就需要使用外鍵來(lái)引用自身锄码。示例代碼如下:
class Comment(models.Model):
content = models.TextField()
origin_comment = models.ForeignKey('self',on_delete=models.CASCADE,null=True)
# 或者
# origin_comment = models.ForeignKey('Comment',on_delete=models.CASCADE,null=True)
外鍵刪除操作:
如果一個(gè)模型使用了外鍵夺英。那么在對(duì)方那個(gè)模型被刪掉后晌涕,該進(jìn)行什么樣的操作∏雉铮可以通過(guò)on_delete
來(lái)指定渐排。可以指定的類(lèi)型如下:
-
CASCADE
:級(jí)聯(lián)操作灸蟆。如果外鍵對(duì)應(yīng)的那條數(shù)據(jù)被刪除了驯耻,那么這條數(shù)據(jù)也會(huì)被刪除。 -
PROTECT
:受保護(hù)炒考。即只要這條數(shù)據(jù)引用了外鍵的那條數(shù)據(jù)可缚,那么就不能刪除外鍵的那條數(shù)據(jù)。 -
SET_NULL
:設(shè)置為空斋枢。如果外鍵的那條數(shù)據(jù)被刪除了帘靡,那么在本條數(shù)據(jù)上就將這個(gè)字段設(shè)置為空。如果設(shè)置這個(gè)選項(xiàng)瓤帚,前提是要指定這個(gè)字段可以為空描姚。 -
SET_DEFAULT
:設(shè)置默認(rèn)值。如果外鍵的那條數(shù)據(jù)被刪除了戈次,那么本條數(shù)據(jù)上就將這個(gè)字段設(shè)置為默認(rèn)值轩勘。如果設(shè)置這個(gè)選項(xiàng),前提是要指定這個(gè)字段一個(gè)默認(rèn)值怯邪。 -
SET()
:如果外鍵的那條數(shù)據(jù)被刪除了绊寻。那么將會(huì)獲取SET
函數(shù)中的值來(lái)作為這個(gè)外鍵的值。SET
函數(shù)可以接收一個(gè)可以調(diào)用的對(duì)象(比如函數(shù)或者方法)悬秉,如果是可以調(diào)用的對(duì)象澄步,那么會(huì)將這個(gè)對(duì)象調(diào)用后的結(jié)果作為值返回回去。 -
DO_NOTHING
:不采取任何行為和泌。一切全看數(shù)據(jù)庫(kù)級(jí)別的約束村缸。
以上這些選項(xiàng)只是Django級(jí)別的,數(shù)據(jù)級(jí)別依舊是RESTRICT武氓!