天天看點

django 1.8 官方文檔翻譯: 2-3-2 關聯對象參考關聯對象參考

Django 文檔協作翻譯小組人手緊缺,有興趣的朋友可以加入我們,完全公益性質。

交流群:467338606

網站:

http://python.usyiyi.cn/django/index.html

關聯對象參考

class RelatedManager

“關聯管理器”是在一對多或者多對多的關聯上下文中使用的管理器。它存在于下面兩種情況:

ForeignKey關系的“另一邊”。像這樣:

from django.db import models

class Reporter(models.Model):
    # ...
    pass

class Article(models.Model):
    reporter = models.ForeignKey(Reporter)           

在上面的例子中,管理器reporter.article_set擁有下面的方法。

ManyToManyField關系的兩邊:

class Topping(models.Model):
    # ...
    pass

class Pizza(models.Model):
    toppings = models.ManyToManyField(Topping)           

這個例子中,topping.pizza_set 和pizza.toppings都擁有下面的方法。

add(obj1[, obj2, …])

把指定的模型對象添加到關聯對象集中。

例如:

>>> b = Blog.objects.get(id=1)
>>> e = Entry.objects.get(id=234)
>>> b.entry_set.add(e) # Associates Entry e with Blog b.           

在上面的例子中,對于ForeignKey關系,e.save()由關聯管理器調用,執行更新操作。然而,在多對多關系中使用add()并不會調用任何 save()方法,而是由QuerySet.bulk_create()建立關系。如果你需要在關系被建立時執行一些自定義的邏輯,請監聽m2m_changed信号。

create(**kwargs)

建立一個新的對象,儲存對象,并将它添加到關聯對象集之中。傳回新建立的對象:

>>> b = Blog.objects.get(id=1)
>>> e = b.entry_set.create(
...     headline='Hello',
...     body_text='Hi',
...     pub_date=datetime.date(2005, 1, 1)
... )

# No need to call e.save() at this point -- it's already been saved.           

這完全等價于(不過更加簡潔于):

>>> b = Blog.objects.get(id=1)
>>> e = Entry(
...     blog=b,
...     headline='Hello',
...     body_text='Hi',
...     pub_date=datetime.date(2005, 1, 1)
... )
>>> e.save(force_insert=True)           

要注意我們并不需要指定模型中用于定義關系的關鍵詞參數。在上面的例子中,我們并沒有傳入blog參數給create()。Django會明白新的 Entry對象blog 應該添加到b中。

remove(obj1[, obj2, …])

從關聯對象集中移除執行的模型對象:

>>> b = Blog.objects.get(id=1)
>>> e = Entry.objects.get(id=234)
>>> b.entry_set.remove(e) # Disassociates Entry e from Blog b.           

和add()相似,上面的例子中,e.save()可會執行更新操作。但是,多對多關系上的remove(),會使用QuerySet.delete()删除關系,意思是并不會有任何模型調用save()方法:如果你想在一個關系被删除時執行自定義的代碼,請監聽m2m_changed信号。

對于ForeignKey對象,這個方法僅在null=True時存在。如果關聯的字段不能設定為None (NULL),則這個對象在添加到另一個關聯之前不能移除關聯。在上面的例子中,從b.entry_set()移除e等價于讓e.blog = None,由于blog的ForeignKey沒有設定null=True,這個操作是無效的。

對于ForeignKey對象,該方法接受一個bulk參數來控制它如果執行操作。如果為True(預設值),QuerySet.update()會被使用。而如果bulk=False,會在每個單獨的模型執行個體上調用save()方法。這會觸發pre_save和post_save,它們會消耗一定的性能。

clear()

從關聯對象集中移除一切對象。

>>> b = Blog.objects.get(id=1)
>>> b.entry_set.clear()           

注意這樣不會删除對象 —— 隻會删除他們之間的關聯。

就像 remove() 方法一樣,clear()隻能在 null=True的ForeignKey上被調用,也可以接受bulk關鍵詞參數。

注意

注意對于所有類型的關聯字段,add()、create()、remove()和clear()都會馬上更新資料庫。換句話說,在關聯的任何一端,都不需要再調用save()方法。

同樣,如果你再多對多關系中使用了中間模型,一些關聯管理的方法會被禁用。

直接指派

通過指派一個新的可疊代的對象,關聯對象集可以被整體替換掉。

>>> new_list = [obj1, obj2, obj3]
>>> e.related_set = new_list           

如果外鍵關系滿足null=True,關聯管理器會在添加new_list中的内容之前,首先調用clear()方法來解除關聯集中一切已存在對象的關聯。否則, new_list中的對象會在已存在的關聯的基礎上被添加。