说说 Python Django 模型之间多对一关联关系

Django 模型种可以定义三种最常见的数据库关联关系:多对一,多对多,一对一。我们先来讲讲多对一关联关系。

1 定义

使用 django.db.models.ForeignKey 类,就可以定义出一个多对一的关联关系。在模型中,添加一个值,作为ForeignKey 类的实例。 ForeignKey 类有一个入参,用于定义想要关联的模型类名。

例如,一个出版社(Press),会出版很多种类的书(Book),而每种书仅来自于一个出版社。那么书与出版社之间,就是多对一的关系:

class Press(models.Model):
    name = models.CharField(max_length=50)


class Book(models.Model):
    name = models.CharField('书名', max_length=50, primary_key=True)
    press = models.ForeignKey(Press, on_delete=models.CASCADE,default='1')

makemigrations 初始化 py 脚本后,用 sqlmigrate 指令可以看出新建了 Press 模型表,并在 Book 模型表中,新建了一个外键。

--
-- Create model Press
--
CREATE TABLE `chart_press` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `name` varchar(50) NOT NULL);
--
-- Add field press to book
--
ALTER TABLE `chart_book` ADD COLUMN `press_id` integer DEFAULT 1 NOT NULL , ADD CONSTRAINT `chart_book_press_id_3e2ac00a_fk_chart_press_id` FOREIGN KEY (`press_id`) REFERENCES `cha
rt_press`(`id`);
ALTER TABLE `chart_book` ALTER COLUMN `press_id` DROP DEFAULT;

这里 Book 模型事先已经新建好。

执行 migrate 指令时,如果抛出 django.db.utils.IntegrityError: (1452, 'Cannot add or update a child row: a foreign key constraint fails (django.#sql-814_45, CONSTRAINT chart_book_press_id_3e2ac00a_fk_chart_ press_id FOREIGN KEY (press_id) REFERENCES chart_press (id))') 错误,说明需要建外键的主表还有数据。
可以清理后,再执行 migrate 指令。

执行 migrate 指令后,可以看到 book 表已经建好了外键:


也可以使用以下方法,创建自关联关系:

models.ForeignKey('self', on_delete=models.CASCADE)

如果需要关联的模型还未定义好,那么可以先定义名称,然后在 ForeignKey 方法中引用该名称,比如上例:

class Press(models.Model):
    pass


class Book(models.Model):
    name = models.CharField('书名', max_length=50, primary_key=True)
    press = models.ForeignKey('Press', on_delete=models.CASCADE,default='1')

2 使用方法

为了让打印更人性化,我们为每个类都重新定义了 __str__ 方法:

class Press(models.Model):
    ...

    def __str__(self):
        return '%s' %(self.name)


class Book(models.Model):
   ...
    def __str__(self):
        return '%s' % (self.name)

2.1 创建模型时,关联对象

在 shell 中,创建了一个出版社,并在新建的书模型中做了关联:


可以在 Book 对象中访问所关联的 Press 对象。

也可以不执行 save() 保存模型,只要该模型被作为其它模型的外键被引用创建,也会被正常保存:

press=Press.objects.create(name='上海文艺出版社')
book=Book.objects.create(name='猫的桌子',press=press)

2.2 在已有模型上,关联对象

也可以先建立 Press 模型实例,然后再关联多个 Book 模型实例。

shell 执行情况:

如果添加错误类型的对象,那么会引发 TypeError!

2.3 查询

xx_set 与 X.objects 对象都支持 filter 方法,方法内可以使用双下划线分隔语法,来获取过滤出需要的数据:

也可以在 filter 方法中,定义多个条件,这些条件将会在 SQL 语句的 where 中以 and 形式连接起来:

Book.objects.filter(press__name='北京联合出版公司',press__book__pages__gt=1)

运行结果:

<QuerySet [<Book: 钢琴的重量>]>

也可以把相关对象,作为参数传入 filter 方法,进行查询:

还可以使用查询集作为参数传入 filter 方法,进行查询:


之前是以 Press 对象为条件,查询出对应的 Book 对象的。我们还可以反向查询,即以 Book 对象为条件,查询出对应的 Press 对象:

因为之前定义的 Book 模型是以 name 作为主键的,所以这里的第一个示例,是以书名作为入参的。

可以这样查询记录数:


2.4 级联删除

因为之前为 Book 模型的 press 字段设置了级联删除,所以如果删除了一个出版社(Press 实例),那么所对应的书(Book 实例),也会被一并删除。

也可以在 filter 方法之后调用删除方法:

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

推荐阅读更多精彩内容