Django中select_related 的作用

Django中这句代码Dynamic.objects.select_related('song')是什么意思?

在 Django 中,这句代码:

复制代码
Dynamic.objects.select_related('song')

的作用是 在查询 Dynamic 模型的同时,预加载 song 关联的外键对象,从而减少数据库查询次数,提高查询效率。


select_related() 适用于 外键(ForeignKey)或一对一(OneToOneField) 的预加载,Django 会使用 SQL 的 JOIN 语句 一次性获取相关数据,避免 N+1 查询问题。


2. 假设的模型结构

假设 Dynamic 模型中有一个 song 字段,它是指向 Song 模型的 外键

复制代码
class Song(models.Model):
    title = models.CharField(max_length=100)
    artist = models.CharField(max_length=100)

class Dynamic(models.Model):
    song = models.ForeignKey(Song, on_delete=models.CASCADE)
    views = models.IntegerField()

如果你直接查询:

复制代码
dynamics = Dynamic.objects.all()
for dynamic in dynamics:
    print(dynamic.song.title)  # 这里会触发 N+1 查询问题

Django 默认是 懒加载 ,每次访问 dynamic.song 时,都会执行一个新的 SQL 查询(N+1 查询问题)。


3. select_related('song') 的优化

复制代码
dynamics = Dynamic.objects.select_related('song')
for dynamic in dynamics:
    print(dynamic.song.title)  # 不会触发额外查询

Django 会使用 JOIN 语句 一次性获取 DynamicSong 表的数据:

复制代码
SELECT dynamic.*, song.*
FROM dynamic
JOIN song ON dynamic.song_id = song.id;

这样,所有数据都在 一次查询 中获取,避免了多个查询,提高了性能。


  • select_related('song') 适用于 外键(ForeignKey)和一对一(OneToOneField) ,使用 JOIN 查询 直接获取数据。
  • prefetch_related('song') 适用于 多对多(ManyToManyField)或反向外键(related_name) ,会执行 两次查询,然后在 Python 代码中进行匹配。

如果 song 是外键,推荐使用 select_related('song'),因为它更高效。

相关推荐
倔强的石头_8 小时前
kingbase备份与恢复实战(二)—— sys_dump库级逻辑备份与恢复(Windows详细步骤)
数据库
jiayou642 天前
KingbaseES 实战:深度解析数据库对象访问权限管理
数据库
李广坤2 天前
MySQL 大表字段变更实践(改名 + 改类型 + 改长度)
数据库
markfeng83 天前
Python+Django+H5+MySQL项目搭建
python·django
爱可生开源社区3 天前
2026 年,优秀的 DBA 需要具备哪些素质?
数据库·人工智能·dba
随逸1774 天前
《从零搭建NestJS项目》
数据库·typescript
加号34 天前
windows系统下mysql多源数据库同步部署
数据库·windows·mysql
シ風箏4 天前
MySQL【部署 04】Docker部署 MySQL8.0.32 版本(网盘镜像及启动命令分享)
数据库·mysql·docker
李慕婉学姐4 天前
Springboot智慧社区系统设计与开发6n99s526(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
数据库·spring boot·后端
QQ4022054964 天前
Python+django+vue3预制菜半成品配菜平台
开发语言·python·django