Django - 回顾(2)- 中介模型

一、中介模型

  我们之前学习图书管理系统时,设计了Publish、Book、Author、AuthorDetail这样几张表,其中Book表和Author表是多对多关系,处理类似这样简单的多对多关系时,使用标准的ManyToManyField就可以了。但是,有时你可能需要关联数据到两个模型之间的关系上。

  例如,有这样一个应用,它记录音乐家所属的音乐小组。我们可以用一个ManyToManyField 表示小组和成员之间的多对多关系。但是,有时你可能想知道更多成员关系的细节,比如成员是何时加入小组的。

  对于这些情况,Django 允许你指定一个中介模型来定义多对多关系。 你可以将其他字段放在中介模型里面。原模型的ManyToManyField 字段将使用through 参数指向中介模型。对于上面的音乐小组的例子,代码如下:

from django.db import models

  class Person(models.Model):
      name = models.CharField(max_length=128)

      def __str__(self):  # __unicode__ on Python 2
          return self.name


  class Group(models.Model):
      name = models.CharField(max_length=128)
      members = models.ManyToManyField("Person", through='Membership')

      def __str__(self):  # __unicode__ on Python 2
          return self.name


  class Membership(models.Model):
      person = models.ForeignKey("Person", on_delete=models.CASCADE)
      group = models.ForeignKey("Group", on_delete=models.CASCADE)
      date_joined = models.DateField()
      invite_reason = models.CharField(max_length=64)

 既然你已经设置好ManyToManyField 来使用中介模型(在这个例子中就是Membership),接下来你要开始创建多对多关系。你要做的就是创建中介模型的实例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
>>> ringo  =  Person.objects.create(name = "Ringo Starr" )    # 创建一个音乐家ringo
>>> paul  =  Person.objects.create(name = "Paul McCartney" # 创建一个音乐家paul
>>> beatles  =  Group.objects.create(name = "The Beatles" # 创建一个音乐小组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 = [...])来创建关系:

1
2
3
4
5
6
# 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模型中所需要的所有信息(比如date_joined和invite_reason);而简单的add、create 和赋值语句是做不到这一点的。所以它们不能在使用中介模型的多对多关系中使用。此时,唯一的办法就是创建中介模型的实例。

  remove()方法被禁用也是出于同样的原因。但是clear()方法却是可用的。它可以清空某个实例所有的多对多关系:

1
2
3
4
5
# Beatles have broken up
>>> beatles.members.clear()
# Note that this deletes the intermediate model instances
>>> Membership.objects. all ()
[]

转载于:https://www.cnblogs.com/wxj1129549016/p/10032198.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值