首頁 > 軟體

Django資料對映(一對一,一對多,多對多)

2022-08-10 18:02:02

資料表關聯關係對映

常用的表關聯方式有三種:

一對一對映
如: 一個身份證對應一個人

一對多對映
如: 一個班級可以有多個學生

多對多對映
如: 一個學生可以報多個課程,一個課程可以有多個學生學習

一對一對映

  • 一對一是表示現實事物間存在的一對一的對應關係。
  • 如:一個家庭只有一個戶主,一個男人有一個妻子,一個人有一個唯一的指紋資訊等

語法

class A(model.Model):
    ...
class B(model.Model):
    屬性 = models.OneToOneField(A, on_delete=xxx)

外來鍵類欄位選項

特殊欄位引數【必須項】:

on_delete

  • models.CASCADE 級聯刪除。 Django模擬SQL約束ON DELETE CASCADE的行為,並刪除包含ForeignKey的物件。
  • models.PROTECT 丟擲ProtectedError 以阻止被參照物件的刪除;[等同於mysql預設的RESTRICT]
  • models.SET_NULL 設定ForeignKey null;需要指定null=True
  • models.SET_DEFAULT 將ForeignKey設定為其預設值;必須設定ForeignKey的預設值。
  • … 其它參請參考檔案 https://docs.djangoproject.com/en/2.2/ref/models/fields/

其餘常用的欄位選項【非必須項】;如:

  • null
  • unique 等

用法範例

建立作家和作家妻子類

# file : xxxxxxxx/models.py
from django.db import models

class Author(models.Model):
    '''作家模型類'''
    name = models.CharField('作家', max_length=50)

class Wife(models.Model):
    '''作家妻子模型類'''
    name = models.CharField("妻子", max_length=50)
    author = models.OneToOneField(Author, on_delete=models.CASCADE)  # 增加一對一屬性 

建立一對一的資料記錄

from .models import *
author1 = Author.objects.create(name='王老師')
wife1 = Wife.objects.create(name='王夫人', author=author1)  # 關聯王老師
author2 = Author.objects.create(name='小澤老師')  # 一對一可以沒有資料對應的資料 

資料查詢

正向查詢

直接通過關聯屬性查詢即可

# 通過 wife 找 author
from .models import Wife
wife = Wife.objects.get(name='王夫人')
print(wife.name, '的老公是', wife.author.name)

反向查詢

  • 通過反向關聯屬性查詢
  • 反向關聯屬性為範例物件.參照類名(小寫),如作家的反向參照為作家物件.wife
  • 當反向參照不存在時,則會觸發異常
# 通過 author.wife 關聯屬性 找 wife,如果沒有對應的wife則觸發異常
author1 = Author.objects.get(name='王老師')
print(author1.name, '的妻子是', author1.wife.name)
author2 = Author.objects.get(name='小澤老師')
try:
    print(author2.name, '的妻子是', author2.wife.name)
except:
    print(author2.name, '還沒有妻子')

一對多對映

  • 一對多是表示現實事物間存在的一對多的對應關係。
  • 如:一個學校有多個班級,一個班級有多個學生, 一本圖書只能屬於一個出版社,一個出版社允許出版多本圖書

語法

當一個A類物件可以關聯多個B類物件時

class A(model.Model):
    ...

class B(model.Model):
    屬性 = models.ForeignKey("一"的模型類, on_delete=xx)

用法範例

有二個出版社對應五本書的情況.

清華大學出版社 有書

  • C++
  • Java
  • Python

北京大學出版社 有書

  • 西遊記
  • 水滸

建立模型類

# file: otm/models.py
from django.db import models

class Publisher(models.Model):
    '''出版社【一】'''
    name = models.CharField('名稱', max_length=50, unique=True)

class Book(models.Model):
    '''書【多】'''
    title = models.CharField('書名', max_length=50)
    publisher = ForeignKey(Publisher, on_delete=models.CASCADE)

建立資料

#先建立 '一' ,再建立 '多'
from .models import *
pub1 = Publisher.objects.create(name='清華大學出版社')
Book.objects.create(title='C++', publisher=pub1)
Book.objects.create(title='Java', publisher_id=1)

#高階建立 - 利用 反向屬性
pub2 = Publisher.objects.create(name='北京大學出版社')
pub2.book_set.create(title='西遊記')

資料查詢

通過 Book 查詢 Publisher【正向】

通過 publisher 屬性查詢即可
​​​​​​​book.publisher
 
abook = Book.objects.get(id=1)
print(abook.title, '的出版社是:', abook.publisher.name)

通過 Publisher 查詢 對應的所有的 Book 【反向】

Django會在Publisher中增加一個屬性來表示對對應的Book們的查詢參照
屬性:book_set  等價於 objects

# 通過出版社查詢對應的書
pub1 = Publisher.objects.get(name='清華大學出版社')
books = pub1.book_set.all()  # 通過book_set 獲取pub1對應的多個Book資料物件
#books = Book.objects.filter(publisher=pub1)  # 也可以採用此方式獲取
print("清華大學出版社的書有:")
for book in books:
   print(book.title)

多對多對映

多對多表達物件之間多對多複雜關係,如: 每個人都有不同的學校(小學,初中,高中,…),每個學校都有不同的學生…
語法

在關聯的兩個類中的任意一個類中,增加:

屬性 = models.ManyToManyField(MyModel)

用法範例

  • 一個作者可以出版多本圖書
  • 一本圖書可以被多名作者同時編寫
class Author(models.Model):
    ...

class Book(models.Model):
    ...
    authors = models.ManyToManyField(Author)

建立模型類

class Author(models.Model):
    '''作家模型類'''
    name = models.CharField('作家', max_length=50)
    def __str__(self):
        return self.name
    
class Book(models.Model):
    '''書模型類'''
    title = models.CharField('書名', max_length=50)
    authors = models.ManyToManyField(Author)
    def __str__(self):
           return self.title 

建立資料

方案1 先建立 author 再關聯 book
    author1 = Author.objects.create(name='呂老師')
    author2 = Author.objects.create(name='王老師')
    # 呂老師和王老師同時寫了一本Python
    book11 = author1.book_set.create(title="Python")
    author2.book_set.add(book11) 
    
方案2 先建立 book 再關聯 author
    book = Book.objects.create(title='python1')
    #郭小鬧和呂老師都參與了 python1 的 創作
    author3 = book.authors.create(name='guoxiaonao')
    book.authors.add(author1)

資料查詢

通過 Book 查詢對應的所有的 Author【正向】

book.authors.all() -> 獲取 book 對應的所有的author的資訊
book.authors.filter(age__gt=80) -> 獲取book對應的作者中年齡大於80歲的作者的資訊

通過 Author 查詢對應的所有的Book【反向】

Django會生成一個反向屬性 book_set 用於表示對對應的book的查詢物件相關操作

author.book_set.all()
author.book_set.filter()

到此這篇關於Django資料對映(一對一,一對多,多對多)的文章就介紹到這了,更多相關Django 資料對映內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com