Django中select_related和prefetch_related的用法与区别详解

2023-11-21 08:59

本文主要是介绍Django中select_related和prefetch_related的用法与区别详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

django-debug-toolbar的安装

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. 使用

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

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 代码

1

2

3

4

5

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 代码

1

2

3

4

5

6

7

8

9

10

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 其他常用用法

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

# 获取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 常用的案例

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

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中select_related和prefetch_related的用法与区别详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/401300

相关文章

kotlin中const 和val的区别及使用场景分析

《kotlin中const和val的区别及使用场景分析》在Kotlin中,const和val都是用来声明常量的,但它们的使用场景和功能有所不同,下面给大家介绍kotlin中const和val的区别,... 目录kotlin中const 和val的区别1. val:2. const:二 代码示例1 Java

CSS Padding 和 Margin 区别全解析

《CSSPadding和Margin区别全解析》CSS中的padding和margin是两个非常基础且重要的属性,它们用于控制元素周围的空白区域,本文将详细介绍padding和... 目录css Padding 和 Margin 全解析1. Padding: 内边距2. Margin: 外边距3. Padd

CSS will-change 属性示例详解

《CSSwill-change属性示例详解》will-change是一个CSS属性,用于告诉浏览器某个元素在未来可能会发生哪些变化,本文给大家介绍CSSwill-change属性详解,感... will-change 是一个 css 属性,用于告诉浏览器某个元素在未来可能会发生哪些变化。这可以帮助浏览器优化

Python基础文件操作方法超详细讲解(详解版)

《Python基础文件操作方法超详细讲解(详解版)》文件就是操作系统为用户或应用程序提供的一个读写硬盘的虚拟单位,文件的核心操作就是读和写,:本文主要介绍Python基础文件操作方法超详细讲解的相... 目录一、文件操作1. 文件打开与关闭1.1 打开文件1.2 关闭文件2. 访问模式及说明二、文件读写1.

详解C++中类的大小决定因数

《详解C++中类的大小决定因数》类的大小受多个因素影响,主要包括成员变量、对齐方式、继承关系、虚函数表等,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录1. 非静态数据成员示例:2. 数据对齐(Padding)示例:3. 虚函数(vtable 指针)示例:4. 继承普通继承虚继承5.

前端高级CSS用法示例详解

《前端高级CSS用法示例详解》在前端开发中,CSS(层叠样式表)不仅是用来控制网页的外观和布局,更是实现复杂交互和动态效果的关键技术之一,随着前端技术的不断发展,CSS的用法也日益丰富和高级,本文将深... 前端高级css用法在前端开发中,CSS(层叠样式表)不仅是用来控制网页的外观和布局,更是实现复杂交

Linux换行符的使用方法详解

《Linux换行符的使用方法详解》本文介绍了Linux中常用的换行符LF及其在文件中的表示,展示了如何使用sed命令替换换行符,并列举了与换行符处理相关的Linux命令,通过代码讲解的非常详细,需要的... 目录简介检测文件中的换行符使用 cat -A 查看换行符使用 od -c 检查字符换行符格式转换将

揭秘Python Socket网络编程的7种硬核用法

《揭秘PythonSocket网络编程的7种硬核用法》Socket不仅能做聊天室,还能干一大堆硬核操作,这篇文章就带大家看看Python网络编程的7种超实用玩法,感兴趣的小伙伴可以跟随小编一起... 目录1.端口扫描器:探测开放端口2.简易 HTTP 服务器:10 秒搭个网页3.局域网游戏:多人联机对战4.

Springboot @Autowired和@Resource的区别解析

《Springboot@Autowired和@Resource的区别解析》@Resource是JDK提供的注解,只是Spring在实现上提供了这个注解的功能支持,本文给大家介绍Springboot@... 目录【一】定义【1】@Autowired【2】@Resource【二】区别【1】包含的属性不同【2】@

详解C#如何提取PDF文档中的图片

《详解C#如何提取PDF文档中的图片》提取图片可以将这些图像资源进行单独保存,方便后续在不同的项目中使用,下面我们就来看看如何使用C#通过代码从PDF文档中提取图片吧... 当 PDF 文件中包含有价值的图片,如艺术画作、设计素材、报告图表等,提取图片可以将这些图像资源进行单独保存,方便后续在不同的项目中使