select_related和prefetch_related的用法与区别

0. 本文借助django-debug-toolbar来展现效果

(19条消息) django-debug-toolbar的安装_骑台风走的博客-CSDN博客icon-default.png?t=M7J4https://blog.csdn.net/qq_52385631/article/details/126695534?spm=1001.2014.3001.5501

1. 介绍

select_related:

将会根据外键关系(注意: 仅限单对单和单对多关系),在执行查询语句的时候通过创建一条包含SQL inner join操作的SELECT语句来一次性获得主对象及相关对象的信息

prefetch_related

对于多对多字段,你不能使用select_related方法,这样做是为了避免对多对多字段执行JOIN操作从而造成最后的表非常大。

Django提供了prefect_related方法来解决这个问题。

prefect_related可用于多对多关系字段,也可用于反向外键关系(related_name)。

相同点:

都作用于queryset对象上面

注意点:

  • 对与单对单或单对多外键ForeignKey字段,使用select_related方法

  • 对于多对多字段和反向外键关系,使用prefetch_related方法

  • 两种方法均支持双下划线指定需要查询的关联对象的字段名

  • 使用Prefetch方法可以给prefetch_related方法额外添加额外条件和属性。

2. 使用

from django.db import models


class UserInfo(models.Model):
    username = models.CharField(verbose_name='用户名', max_length=225)

    def __str__(self):
        return self.username


class Tag(models.Model):
    name = models.CharField(verbose_name='标签名称', max_length=225)

    def __str__(self):
        return self.name


class Article(models.Model):
    title = models.CharField(verbose_name='标题', max_length=225)
    content = models.CharField(verbose_name='内容', max_length=225)
    # 外键
    username = models.ForeignKey(verbose_name='用户', to='UserInfo', on_delete=models.DO_NOTHING)
    tag = models.ManyToManyField(verbose_name='标签', to='Tag')

    def __str__(self):
        return self.title

2.1 原生的查询

2.1.1 代码

def article_list(request):
    if request.method == 'GET':
        # select_related---->queryset
        article_queryset = models.Article.objects.all()
        return render(request, 't2.html', context={'article_queryset': article_queryset})

2.1.2 图示

2.1.3 查询解释 

1.从图示我们可以看出来,一共进行13次查询,且有10次重复的!!!

原因是:当我们第一次查询时,返回的值,只有文章对象,对于标签以及用户,并没有查询,当前端界面需要这两个时,每循环一次,就会去数据库查询一次

2.为了避免重复查询,django提供select_related和prefetch_related方法来提升数据库查询效率,类似于SQL的JOIN方法。

3.效果就是当第一次查询时,进行连表,一次性把所有数据全部查询到

2.2 使用select_related

2.2.2 代码

from django.shortcuts import render

from blog import models


def article_list(request):
    if request.method == 'GET':
        # select_related---->queryset
        article_queryset = models.Article.objects.all().select_related('tag', 'username')
        return render(request, 't2.html', context={'article_queryset': article_queryset})

2.2.3 图示

 2.2.4 解释

可以看到现在只有三次查询,耗时大大减少

 2.2.5 其他常用用法

# 获取id=1的文章对象同时,获取其相关username信息
Article.objects.select_related('username').get(id=1)

# 获取id=1的文章对象同时,获取其相关作者名字信息
Article.objects.select_related('username__username').get(id=1)

# 获取id=1的文章对象同时,获取其相关tag和相关作者名字信息。下面方法等同。
# 方式一:
Article.objects.select_related('tag', 'username__username').get(id=1)
# 方式二:
Article.objects.select_related('tag').select_related('username__username').get(id=1)

# 使用select_related()可返回所有相关主键信息。all()非必需。
Article.objects.all().select_related()

# 获取Article信息同时获取username信息。filter方法和selected_related方法顺序不重要。
# 方式一:
Article.objects.filter(tag__gt=3).select_related('username')
# 方式二:
Article.objects.select_related('username').filter(tag__gt=3)

2.3. 使用prefetch_related方法

对于多对多字段,你不能使用select_related方法,这样做是为了避免对多对多字段执行JOIN操作从而造成最后的表非常大。

2.3.1 常用的案例

articles = Article.objects.all().select_related('category').prefecth_related('tags')

# 文章列表及每篇文章的tags对象名字信息
Article.objects.all().prefetch_related('tags__name')

# 获取id=13的文章对象同时,获取其相关tags信息
Article.objects.prefetch_related('tags').get(id=13)

# 获取文章列表及每篇文章相关的名字以P开头的tags对象信息
Article.objects.all().prefetch_related(
    Prefetch('tags', queryset=Tag.objects.filter(name__startswith="P"))
)

# 文章列表及每篇文章的名字以P开头的tags对象信息, 放在article_p_tag列表
Article.objects.all().prefetch_related(
    Prefetch('tags', queryset=Tag.objects.filter(name__startswith="P")),
to_attr='article_p_tag'
)

本文参考

Django基础(29): select_related和prefetch_related的用法与区别 (qq.com)

https://cn.bing.com/ck/a?!&&p=e0abbfe758bda72aJmltdHM9MTY2MjI0OTYwMCZpZ3VpZD0wZDM2NTliMy1hZDc5LTYwY2YtMDk4Mi00YmI2YWM1NzYxNmMmaW5zaWQ9NTEzNA&ptn=7&hsh=3&fclid=0d3659b3-ad79-60cf-0982-4bb6ac57616c&u=a1aHR0cHM6Ly93d3cuY25ibG9ncy5jb20vY2NvcnovcC82MDk3MTcxLmh0bWw&ntb=1 https://cn.bing.com/ck/a?!&&p=f1ee206eac3dcc27JmltdHM9MTY2MjI0OTYwMCZpZ3VpZD0wZDM2NTliMy1hZDc5LTYwY2YtMDk4Mi00YmI2YWM1NzYxNmMmaW5zaWQ9NTE1Nw&ptn=7&hsh=3&fclid=0d3659b3-ad79-60cf-0982-4bb6ac57616c&u=a1aHR0cHM6Ly93d3cuY25ibG9ncy5jb20vd2FuZ3l1ZTA5MjUvcC8xMTExODczOS5odG1s&ntb=1

  • 84
    点赞
  • 65
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 73
    评论
### 回答1: `select_related` 是 Django ORM 中的一个方法,用于查询关联模型的数据。它可以避免进行多次数据库查询,从而提高查询效率。 例如,如果你有一个模型 `Author` 和一个模型 `Book`,并且一个作者可以对应多本书,那么你可以使用以下代码来获取一个作者对应的所有书: ``` author = Author.objects.get(pk=1) books = Book.objects.filter(author=author) ``` 这样会导致两次数据库查询,第一次查询获取作者的数据,第二次查询获取书的数据。 如果使用 `select_related`,就可以一次性获取作者和对应的所有书: ``` author = Author.objects.select_related().get(pk=1) books = author.book_set.all() ``` 这样只需要进行一次数据库查询,查询速度会更快。 ### 回答2: select_related是Django ORM的一个功能,用于优化查询性能。它通过使用JOIN操作一次性将多个相关模型的数据取出,避免了多次查询数据库的问题。 在默认情况下,当我们访问一个模型对象的外键或者一对一关联字段时,Django会自动发出相应的数据库查询。这意味着在处理大量对象时,会导致大量查询操作,严重影响性能。 而select_related的作用就是通过指定需要一起查询的相关模型,一次性将这些相关模型的数据取出,避免多次查询。这样,我们就可以在一次数据库访问中获取所有相关模型的数据。使用select_related可以大幅减少数据库的访问次数,提高查询性能。 使用select_related非常简单,只需要在查询时使用select_related()方法,并指定需要查询的相关模型即可。如下所示: Model.objects.select_related('related_model') 其中,Model是主要要查询的模型类,related_model是Model的一个关联字段。使用select_related后,查询结果将包含主要模型和相关模型的所有数据。 需要注意的是,使用select_related查询可能会导致较大的数据集被加载到内存中,如果查询结果集很大,可能会导致性能问题。因此,在使用select_related时,应该根据实际情况谨慎使用,避免查询结果集过大。 综上所述,select_related是Django ORM中的一个优化查询性能的功能,通过一次性将多个相关模型的数据取出,避免了多次查询数据库的问题。使用方法简单,在查询时使用select_related()方法,并指定需要查询的相关模型即可。但需要注意查询结果集的大小,避免性能问题的发生。 ### 回答3: select_related是Django ORM中的一个方法,用于在查询数据库时进行关联查询,减少查询次数,提高查询效率。它可以在一次查询中同时查询多个相关联的表,而不需要多次查询。 select_related方法的使用非常简单,只需要在查询时使用它即可。例如,假设有一个模型A与模型B关联,我们可以通过以下方式使用select_related: A.objects.select_related('B').filter(条件) 这里的条件可以是任意的查询条件,比如过滤某些数据。select_related('B')表示在查询A模型时同时查询与之关联的B模型。 使用select_related方法的好处是,它会执行一次SQL查询,将A模型和B模型的数据一起返回。这样可以避免在使用A模型数据时,每次都去查询关联的B模型数据,从而减少了数据库查询的次数,提高了查询效率。 需要注意的是,select_related只能进行一层的关联查询,不能进行多层的关联。如果要进行多层关联查询,可以使用prefetch_related方法。 另外,使用select_related方法还需谨慎,因为它会将所有关联的数据一起查询出来,如果关联的数据量很大,会占用大量内存。所以,在使用时要根据实际情况考虑是否使用select_related,并合理设计数据库的关联关系。 总之,select_related是Django ORM中非常实用的方法,可以在一次查询中同时查询多个相关联的表,减少数据库查询次数,提高查询效率。但需要注意使用时的内存占用问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 73
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

骑猪去兜风z1

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值