马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。
您需要 登录 才可以下载或查看,没有账号?立即注册
x
在 Django 中,select_related 和 prefetch_related 是用于优化查询的两个紧张方法。它们通过淘汰数据库查询次数来提高性能,但它们的工作方式有所不同。
select_related
select_related 使用 SQL 的 JOIN 操纵来获取相关对象。它实用于一对一和多对一的关系(如 ForeignKey 和 OneToOneField)。它会在单个查询中获取相关对象的数据,从而淘汰数据库查询次数。
示例
假设我们有以下模型:
- 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 来优化查询:
- # 获取所有书籍及其对应的作者
- 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
prefetch_related 使用单独的查询来获取相关对象,然后在 Python 中进行关联。它实用于一对多和多对多的关系(如 ForeignKey、OneToOneField 和 ManyToManyField)。它会在单独的查询中获取相关对象的数据,然后在内存中进行关联。
示例
假设我们有以下模型:
- 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 来优化查询:
- # 获取所有作者及其对应的书籍
- 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 照旧 prefetch_related
- select_related:实用于一对一和多对一的关系。它使用 SQL JOIN 操纵,在单个查询中获取相关对象的数据。
- prefetch_related:实用于一对多和多对多的关系。它使用单独的查询来获取相关对象的数据,然后在内存中进行关联。
组合使用
在某些情况下,可能需要同时使用 select_related 和 prefetch_related 来优化查询。
示例
假设我们有以下模型:
- 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_related 和 prefetch_related 来优化查询:
- # 获取所有书籍及其对应的作者和出版社
- 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') 会在单独的查询中获取作者及其对应的册本,然后在内存中进行关联。
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。 |