處理類似搭配 pizza 和 topping 這樣簡單的多對多關系時,使用標準的ManyToManyField 就可以了。但是,有時你可能需要關聯數據到兩個模型之間的關系上。
例如,有這樣一個應用,它記錄音樂家所屬的音樂小組。我們可以用一個ManyToManyField 表示小組和成員之間的多對多關系。但是,有時你可能想知道更多成員關系的細節,比如成員是何時加入小組的。
對于這些情況,Django 允許你指定一個中介模型來定義多對多關系。 你可以將其他字段放在中介模型里面。源模型的ManyToManyField 字段將使用through 參數指向中介模型。對于上面的音樂小組的例子,代碼如下:
既然你已經設置好ManyToManyField 來使用中介模型(在這個例子中就是Membership),接下來你要開始創建多對多關系。你要做的就是創建中介模型的實例:
ringo = Person.objects.create(name="Ringo Starr")
paul = Person.objects.create(name="Paul McCartney")
beatles = Group.objects.create(name="The Beatles")
m1 = Membership(person=ringo, group=beatles,
... date_joined=date(1962, 8, 16),
... invite_reason="Needed a new drummer.")
m1.save()
beatles.members.all()
[<Person: Ringo Starr>]
ringo.group_set.all()
[<Group: The Beatles>]
m2 = Membership.objects.create(person=paul, group=beatles,
... date_joined=date(1960, 8, 1),
... invite_reason="Wanted to form a band.")
beatles.members.all()
[<Person: Ringo Starr>, <Person: Paul McCartney>]
與普通的多對多字段不同,你不能使用add、 create和賦值語句(比如,beatles.members = [...])來創建關系:
THIS WILL NOT WORK
beatles.members.add(john)
NEITHER WILL THIS
beatles.members.create(name="George Harrison")
AND NEITHER WILL THIS
beatles.members = [john, paul, ringo, george]
為什么不能這樣做? 這是因為你不能只創建 Person和 Group之間的關聯關系,你還要指定 Membership模型中所需要的所有信息;而簡單的add、create 和賦值語句是做不到這一點的。所以它們不能在使用中介模型的多對多關系中使用。此時,唯一的辦法就是創建中介模型的實例。
remove()方法被禁用也是出于同樣的原因。但是clear() 方法卻是可用的。它可以清空某個實例所有的多對多關系:
Beatles have broken up
beatles.members.clear()
Note that this deletes the intermediate model instances
Membership.objects.all()
[]