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
, CONSTRAINTchart_book_press_id_3e2ac00a_fk_chart_ press_id
FOREIGN KEY (press_id
) REFERENCESchart_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 方法之后调用删除方法: