说说 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 中,创建了一个出版社,并在新建的书模型中做了关联:

扫描二维码关注公众号,回复: 8861795 查看本文章

可以在 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 方法之后调用删除方法:

发布了614 篇原创文章 · 获赞 689 · 访问量 90万+

猜你喜欢

转载自blog.csdn.net/deniro_li/article/details/104088435