django之select_related 与 prefetch_related用法

在 Django 中,select_relatedprefetch_related 是用于优化查询的两个重要方法。它们通过减少数据库查询次数来提高性能,但它们的工作方式有所不同。

select_related 使用 SQL 的 JOIN 操作来获取相关对象。它适用于一对一和多对一的关系(如 ForeignKey 和 OneToOneField)。它会在单个查询中获取相关对象的数据,从而减少数据库查询次数。

示例

假设我们有以下模型:

python 复制代码
from django.db import models

class Author(models.Model):
    name = models.CharField(max_length=100)
    email = models.EmailField()

    def __str__(self):
        return self.name

class Book(models.Model):
    title = models.CharField(max_length=200)
    publication_date = models.DateField()
    author = models.ForeignKey(Author, related_name='books', on_delete=models.CASCADE)

    def __str__(self):
        return self.title

可以使用 select_related 来优化查询:

python 复制代码
# 获取所有书籍及其对应的作者
books = Book.objects.select_related('author').all()

for book in books:
    print(f"{book.title} by {book.author.name}")

在这个示例中,select_related('author') 会在单个查询中获取所有书籍及其对应的作者,从而避免了每次访问 book.author 时的额外查询。

prefetch_related 使用单独的查询来获取相关对象,然后在 Python 中进行关联。它适用于一对多和多对多的关系(如 ForeignKey、OneToOneField 和 ManyToManyField)。它会在单独的查询中获取相关对象的数据,然后在内存中进行关联。

示例

假设我们有以下模型:

python 复制代码
from django.db import models

class Author(models.Model):
    name = models.CharField(max_length=100)
    email = models.EmailField()

    def __str__(self):
        return self.name

class Book(models.Model):
    title = models.CharField(max_length=200)
    publication_date = models.DateField()
    author = models.ForeignKey(Author, related_name='books', on_delete=models.CASCADE)

    def __str__(self):
        return self.title

可以使用 prefetch_related 来优化查询:

python 复制代码
# 获取所有作者及其对应的书籍
authors = Author.objects.prefetch_related('books').all()

for author in authors:
    print(f"Books by {author.name}:")
    for book in author.books.all():
        print(f" - {book.title}")

在这个示例中,prefetch_related('books') 会在单独的查询中获取所有作者及其对应的书籍,然后在内存中进行关联,从而避免了每次访问 author.books.all() 时的额外查询。

  • select_related适用于一对一和多对一的关系。它使用 SQL JOIN 操作,在单个查询中获取相关对象的数据。
  • prefetch_related适用于一对多和多对多的关系。它使用单独的查询来获取相关对象的数据,然后在内存中进行关联。

组合使用

在某些情况下,可能需要同时使用 select_relatedprefetch_related 来优化查询。

示例

假设我们有以下模型:

python 复制代码
from django.db import models

class Publisher(models.Model):
    name = models.CharField(max_length=100)

    def __str__(self):
        return self.name

class Author(models.Model):
    name = models.CharField(max_length=100)
    email = models.EmailField()
    publisher = models.ForeignKey(Publisher, related_name='authors', on_delete=models.CASCADE)

    def __str__(self):
        return self.name

class Book(models.Model):
    title = models.CharField(max_length=200)
    publication_date = models.DateField()
    author = models.ForeignKey(Author, related_name='books', on_delete=models.CASCADE)

    def __str__(self):
        return self.title

可以同时使用 select_relatedprefetch_related 来优化查询:

python 复制代码
# 获取所有书籍及其对应的作者和出版社
books = Book.objects.select_related('author__publisher').prefetch_related('author__books').all()

for book in books:
    print(f"{book.title} by {book.author.name} (Publisher: {book.author.publisher.name})")
    print("Other books by this author:")
    for other_book in book.author.books.all():
        if other_book != book:
            print(f" - {other_book.title}")

在这个示例中,select_related('author__publisher') 会在单个查询中获取书籍、作者及其对应的出版社,而 prefetch_related('author__books') 会在单独的查询中获取作者及其对应的书籍,然后在内存中进行关联。

相关推荐
FreakStudio2 小时前
一文速通 Python 并行计算:13 Python 异步编程-基本概念与事件循环和回调机制
python·pycharm·协程·多进程·并行计算·异步编程
豌豆花下猫4 小时前
让 Python 代码飙升330倍:从入门到精通的四种性能优化实践
后端·python·ai
夏末蝉未鸣014 小时前
python transformers库笔记(BertForTokenClassification类)
python·自然语言处理·transformer
weixin_418813876 小时前
Python-可视化学习笔记
笔记·python·学习
Danceful_YJ6 小时前
4.权重衰减(weight decay)
python·深度学习·机器学习
Zonda要好好学习7 小时前
Python入门Day5
python
电商数据girl8 小时前
有哪些常用的自动化工具可以帮助处理电商API接口返回的异常数据?【知识分享】
大数据·分布式·爬虫·python·系统架构
CoooLuckly8 小时前
numpy数据分析知识总结
python·numpy
超龄超能程序猿8 小时前
(六)PS识别:源数据分析- 挖掘图像的 “元语言”技术实现
python·组合模式
开开心心_Every8 小时前
全能视频处理工具介绍说明
开发语言·人工智能·django·pdf·flask·c#·音视频