Models and databases 之二 关系

在关系数据库中设置表与表之间的关系

Many-to-one的关系

定义一个多对一的关系,用django.db.models.ForeignKey。像用其他字段类型意向方便。ForeignKey有一个必须的参数:此类关联的model

from diango.db import models
class Manufacturer(models.Model):
      pass
class Car(model.Model):
      manufacturer = models.ForeignKey(Manufacturer,on_delete=models.CASCADE)
>>>
一个与自身有多对一关系的对象
models.ForeignKey('self', on_delete=models.CASCADE)

可以通过Abstract base class与多个子类建立多对一关系
products/models.py

from django.db import models
class AbstractCar(models.Model):
      manufacturer = models.ForeignKey('Maufacturer',on_delete=models.CASCADE)
      class Meta:
                abstract = True

production/models.py

from django.db import models
from products.models import AbstractCar
class Manufacturer(models.Model):
    pass
class Car(AbstractCar):
    pass
class BENSCar(AbstractCar):
    pass
>>>Car.manufacturer will point to `production.Manufacturer` here.

ForeignKey还接受一些其他的参数,用于定义关系。

  • ForeignKey.on_delete
    当一个被引用的对象被删除的时候,Django会通过on_delete的参数模仿SQL的相关的操作
.CASCADE    当被引用的对象被删除的时候,此引用对象也要一并执行删除的操作
.PROTECT    禁止删除并产生一个异常(ProtectedError)
.SET_NULL   把外键设置为null,规划时此字段要设置可接受null
.SET_DEFAULT 把外键设置为默认值,规划是此字段要设置一个默认值

设置其他值 采用一个方法的形式  callable
from django.conf import settings
from django.contrib.auth import get_user_model from django.db import models
def get_sentinel_user():
return get_user_model().objects.get_or_create(username='deleted')[0]
class MyModel(models.Model): user = models.ForeignKey(
settings.AUTH_USER_MODEL,
on_delete=models.SET(get_sentinel_user), )

.DO_NOTHING  什么也不做
  • ForeignKey.limit_choices_to
    限制字段可供选择的范围。参数是字典的形式或者一个Q对象
staff_member = models.ForeignKey(User,on_delete=models.CASCADE,limit_choices_to={'is_staff':True},)
限制了staff_member只有User的is_staff的字段可以作为外键关联的记录
  • ForeignKey.related_name
    关联对象反向引用描述符
    当一张表的多个字段指向同一张表时,会出错。系统无法知道,通过另外一张表,访问XXX_set属性访问到的是哪个属性。这时,我们就需要为每个字段定义一个related_name属性,另外一张表访问这个表时,就会根据related_name的值来得到各个属性了。

Many-to-many

定义一个Many-to-many关系使用ManyToManyField。ManyToManyField有一个比传参数。

from django.db import models
class Topping(models):
    pass
class Pizza(models.Model):
    topping = models.ManyToManyField(Topping)

However, sometimes you may need to associate data with the relationship between two models.
用两个模型的关系模型将此二者的数据联系起来
例如:音乐家和音乐乐团之间的关系。他们是多对多的关系,但是关于这个会员资格的一些信息需要一个额外的数据模型来解释。Django允许指定一个model来管理这种关系,设置一些字段在这个中间模型里。

from django.db import models
class Person(models.Model):
      name = models.CharField(max_length=128)
      def __str__(self):
            return self.name
class Group(models.Model):
       name = models.CharField(max_length=128)
       members = models.ManyToManyField(Person,through='Membership')
        def __str__(self):
            return self.name
class Membership(models.Model):
       person = models.ForeignKey(Person,on_delete=models.CASCADE)
        group = models.ForeignKey(Group,on_detele=models.CASCADE)
        date_joined = models.DateField()
        invite_reason = models.CharField(max_length=64)
>>>其中Membership就是中间model。中间model中ForeignKey把两个有关联的model联系起来
>>> 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()
<QuerySet [<Person: Ringo Starr>]>
>>> ringo.group_set.all()
<QuerySet [<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()
<QuerySet [<Person: Ringo Starr>, <Person: Paul McCartney>]>
在有中间model的多对多的关系中,往常的add(),create(),set()不起作用
>>> # The following statements will not work
>>> beatles.members.add(john)
>>> beatles.members.create(name="George Harrison") 
>>> beatles.members.set([john, paul, ringo, george])
只能通过创建中间Model的实例的方式添加新的关系

在查询的时候,与普通的Many-to-Many的关系一样

# Find all the members of the Beatles that joined after 1 Jan 1961
>>> Person.objects.filter(
... group__name='The Beatles',
... membership__date_joined__gt=date(1961,1,1)) 
<QuerySet [<Person: Ringo Starr]>
>>> ringos_membership = Membership.objects.get(group=beatles, person=ringo) 
>>> ringos_membership.date_joined
datetime.date(1962, 8, 16)
>>> ringos_membership.invite_reason
'Needed a new drummer.'

One-to-one relationships

用OneToOneField来定义一个一对一的关系,需要传递一个把绑定的model

from django.db import models
class Place(models.Model):
      name = models.CharField(max_length=50)
      address = models.CharField(max_length=80)
      def __str__(self):
          return "%s the place" % self.name
class Restaurant(model.Model):
       place = models.OneToOneField(Place,on_delete=models.CASCADE,primary_key=True,)
       serves_hot_dogs = models.BooleanField(default=False)
       serves_pizza = models.BooleanField(default=False)
        def __str__(self):
              return "%s the restaurant"% self.place.name
class Waiter(models.Model):
        restaurant = models.ForeignKey(Restaurant,on_delete=models.CASCADE)
        name = models.CharField(max_length=50)
        def __str__(self):
              return "%s the waiter at %s"%(self.name,self,restaurant)

p1 = Place(name='Demon Dogs',address='994 W. Fullerton')
p1.save()
p2=Place(name='Ace Hardware',address='1013 N. Ashland')
p2.save()
r = Restaurant(place=p1,serves_hot_dogs=True,serves_pizza=False)
r.save()
------
r.place
<Place: Demon Dogs the place>
p1.restaurant   #由于p1与r关联起来了,可以通过p1.resaurant的方式找到p1关联的resaurant。
<Restaurant: Demon Dogs the restaurant>

添加一个Waiter
w = r.waiter_set.create(name='Joe')

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容