DRF从入门到精通六(排序组件、过滤组件、分页组件、异常处理)

文章目录

  • 一、排序组件
    • 继承GenericAPIView使用DRF内置排序组件
    • 继承APIView编写排序
  • 二、过滤组件
    • 继承GenericAPIView使用DRF内置过滤器实现过滤
    • 使用第三方模块django-filter实现and关系的过滤
    • 自定制过滤类
    • 排序搭配过滤使用
  • 三、分页组件
    • 分页器一:Pagination(基本分页)
    • 分页器二:LimitOffsetPagination(偏移分页)
    • 分页器三:CursorPagination(游标分页)
  • 四、异常处理

一、排序组件

一般涉及到了查询所有才有排序,对于表模型查询所有数据时需要根据某个字段进行排序时,我们就可以使用到REST framework提供的内置排序组件OrderingFilter来帮助我们快速指名数据按照指定字段进行排序

使用方法

  1. 导入DRF内置排序组件:from rest_framework.filters import OrderingFilter
  2. 在视图类中设置类属性:filter_backends=[OrderingFilter]
    注意这里能使用这个类属性,只有是继承了GenericAPIView才行。否则无法使用
    所以像继承APIView则无法使用内置过滤器
  3. REST framework会在请求的查询字符串参数中检查是否包含了ordering参数
  4. 如果包含了ordering参数,则按照ordering参数指明排序字段对数据集进行排序
  5. 前端可以传递的ordering参数的可选字段值需要再ordering_fields中指明

继承GenericAPIView使用DRF内置排序组件

	'需要有表数据,我这里直接沿用上一篇博客中的使用,然后使用自动生成路由的方式'from rest_framework.generics import ListAPIViewfrom rest_framework.viewsets import ViewSetMixinfrom rest_framework.filters import OrderingFilterfrom .serializer import BookSerializerfrom . import modelsclass BookView(ViewSetMixin,ListAPIView):queryset = models.Book.objects.all()serializer_class = BookSerializer'配置属性'filter_backends = [OrderingFilter]'必须指定表模型数据字段'ordering_fields = ['price'] # 可以写多个排序字段# ordering_fields = ['price','id']'127.0.0.1:8000/api/v2/books/?ordering=-price 倒序''127.0.0.1:8000/api/v2/books/?ordering=price 升序'

在这里插入图片描述


继承APIView的是使用不了以上DRF提供的排序组件,需要自己写,自己从请求地址中取出排序规则,然后自己排序

继承APIView编写排序

	from rest_framework.views import APIViewfrom rest_framework.response import Responsefrom rest_framework.viewsets import ViewSetMixinfrom rest_framework.mixins import ListModelMixin'这里我还是沿用上面自动生成路由的方式,所以需要配置ViewSetMixin'class BookView(ViewSetMixin, APIView, ListModelMixin):'由于是APIView需要自己重写list方法,在自动生成路由中是get:list'def list(self, request, *args, **kwargs):'从地址栏中取出过滤条件'print(request.query_params)  # ?ordering=-price,idquery_params = request.query_params  # {ordering:price}'''支持多个条件排序ordering=-price,id'''# try:'127.0.0.1:8000/api/v2/books/?ordering=-price,id'if ',' in query_params.get('ordering'):query = query_params.get('ordering').split(',')book_list = models.Book.objects.all().order_by(*query)else:book_list = models.Book.objects.all().order_by(query_params.get('ordering'))ser = BookSerializer(instance=book_list, many=True)return Response(ser.data)

在这里插入图片描述


二、过滤组件

restful规范中,要求请求地址中带过滤条件,五个接口中,只有查询所有接口需要过滤和排序。

其实上面排序使用的就是DRF内置过滤器,因为排序本就是过滤的一种特殊情况,所以这里就不在过多介绍了

继承GenericAPIView使用DRF内置过滤器实现过滤

	from rest_framework.filters import SearchFilterfrom rest_framework.viewsets import ViewSetMixinfrom rest_framework.generics import ListAPIViewfrom rest_framework.filters import OrderingFilter,SearchFilterclass BookView(ViewSetMixin, GenericAPIView, ListModelMixin):queryset = models.Book.objects.all()serializer_class = BookSerializer'SearchFilter,使用的是模糊匹配'filter_backends = [SearchFilter]'''只能填写表中字段'''search_fields = ['name']search_fields = ['name','price']  可多字段过滤# 127.0.0.1:8000/api/v2/books/?search=东# 127.0.0.1:8000/api/v2/books/?search=46  '因为是多字段过滤,并且SearchFilter是模糊匹配,所以如果多字段中有一样的东西,如1都会被查到'

使用DRF内置过滤器,它搜索结果的关系为or(或)的关系,不是and关系。并且只能是使用关键字search才能使用,如果我们想要用name=东&price=66这种方式的得使用别的方法来实现

在这里插入图片描述


使用第三方模块django-filter实现and关系的过滤

	'首先需要安装第三方模块:pip install django-filter'from django_filters.rest_framework import DjangoFilterBackendfrom rest_framework.viewsets import ViewSetMixinfrom rest_framework.generics import ListAPIView'使用这个第三方模块可以实现and关系,并且只能是精准匹配。并且暂不支持or关系'class BookView(ViewSetMixin,ListAPIView):queryset = models.Book.objects.all()serializer_class = BookSerializerfilter_backends = [DjangoFilterBackend]filterset_fields = ['name', 'price']# 127.0.0.1:8000/api/v2/books/?name=东游记&price=46 精准匹配'但是对于django-filter来讲它是支持扩写的,所以是可以支持模糊匹配,具体操作自寻查找'

在这里插入图片描述

使用这个django-filter还是有局限性的,无法实现or关键以及模糊匹配,那么我们可以使用自定制过滤类


自定制过滤类

	使用步骤:1.定义一个过滤类,并且继承BaseFilterBackend2.重写filter_queryset方法,并且在内部完成过滤规则3.在视图类中配置自定义的过滤类from rest_framework.filters import BaseFilterBackendclass CommonFilter(BaseFilterBackend):def filter_queryset(self, request, queryset, view):'''queryset是之前所有数据,models.Book.object.all()'''# request是当次请求# query_params = request.query_params'方式一:'name = request.query_params.get('name',None)price = request.query_params.get('price',None)res = queryset.filter(name__contains=name,price=price)'方式二:'if request.query_params.get('name'):queryset = queryset.filter(name__contains=request.query_params.get('name'))if request.query_params.get('price'):'支持链式调用'queryset = queryset.filter(price=request.query_params.get('price'))return queryset  # 这里返回过滤后的对象 resfrom .filters import CommonFilter  class BookView(ViewSetMixin,ListAPIView):queryset = models.Book.objects.all()serializer_class = BookSerializer'无需再配置字段了,因为在自定制类中已经写好了过滤规则,所以在视图类中无需配置'filter_backends = [CommonFilter]# 127.0.0.1:8000/api/v2/books/?name=游记&price=666'这样就实现了模糊匹配name字段并且精准匹配price字段,以及查询单个字段的规则'

在这里插入图片描述


排序搭配过滤使用

	from .filters import CommonFilter  # 使用的是上面的自定制过滤类from rest_framework.filters import OrderingFilterclass BookView(ViewSetMixin,ListAPIView):queryset = models.Book.objects.all()serializer_class = BookSerializerfilter_backends = [OrderingFilter,CommonFilter]  # 排序加过滤,从左到右依次执行ordering_fields = ['price','id']

在这里插入图片描述


三、分页组件

分页只针对查询所有的接口,其他四个接口不需要分页。drf内置了三个分页器,对应三种分页方式,内置的分页类不能直接使用,需要继承,定制一些参数后才能使用。一个接口只能有一种分页方式,不能混合分页方式

分页器一:Pagination(基本分页)

通过自定义Pagination类,来为视图添加不同分页行为。在视图中通过pagination_clas属性来指明。

	from rest_framework.pagination import PageNumberPaginationclass CommonPageNumberPagination(PageNumberPagination):page_size = 3 # 默认每页显示的条数page_query_param = 'page' # 使用该关键字指定页码(客户端使用)page_size_query_param = 'size' # 通过该关键字可以手动指定页面显示的数据条数(客户端使用)max_page_size = 5 # 只针对客户端手动指定页面显示的数据条数做出一个限制,但不影响page_size属性。

视图类

	'导入配置的分页类'from .pagination import CommonPageNumberPaginationfrom rest_framework.viewsets import ViewSetMixinfrom rest_framework.generics import ListAPIViewfrom . import modelsfrom .serializer import BookSerializerfrom .filters import CommonFilterfrom rest_framework.filters import OrderingFilterclass BookView(ViewSetMixin,ListAPIView):queryset = models.Book.objects.all()serializer_class = BookSerializer'注意分页方式一个接口只能使用一种,所以没有加中括号'pagination_class = CommonPageNumberPagination'''使用分页并不影响排序和过滤'''filter_backends = [OrderingFilter,CommonFilter]filterset_fields = ['name', 'price']# 127.0.0.1:8000/api/v2/books/?page=2# 127.0.0.1:8000/api/v2/books/# 127.0.0.1:8000/api/v2/books/?page=2&size=3

在这里插入图片描述

响应时额外携带了3个参数,分别是:

  1. count:book接口的数据总量
  2. next:下一页的URL
  3. previous:上一页的URL,如果没有上一页返回None

当然,我们也可以手动指定页面显示的条数,根据page_size_query_param属性值来作为关键字。

在这里插入图片描述

可以看到我们需要显示的是6条数据,而后端只响应了5条,这是因为max_page_size属性值的作用:限制了客户端手动获取数据时最大返回的数据条数。


分页器二:LimitOffsetPagination(偏移分页)

偏移分页,该分页组件作用是:从哪一条数据之后开始显示,以及制定数据显示的条数

前端访问形式:http://127.0.0.1:8080/book/?offset=2&limit=4,这表示从第3条数据之后显示4条数据

自定义分页类

	from rest_framework.pagination import LimitOffsetPaginationclass CommonLimitOffsetPagination(LimitOffsetPagination):# 等同于上面的:page_sizedefault_limit = 2  # 如果前端没有手动指定获取的数据条数时,使用该属性值limit_query_param = 'limit'  # 前端通过该关键字指定数据条数'每页显示条数,查询的条数  例子 ?limit=100 意思就是每页显示100条,如果不传应用default_limit的参数'offset_query_param = 'offset'  # 通过该关键字指定从哪条数据之后开始获取数据'偏移量  举个例子offset=6&limit=30  意思就是从第6条开始,拿30条'# 等同于上面的:max_page_sizemax_limit = 5  # 只限制limit_query_param能够获取的最大数据条数,而不影响default_limit

视图类

	'导入配置的分页类'from .pagination import CommonPageNumberPaginationclass BookView(ViewSetMixin,ListAPIView):queryset = models.Book.objects.all()serializer_class = BookSerializer'注意分页方式一个接口只能使用一种,所以没有加中括号'pagination_class = CommonPageNumberPagination# http://127.0.0.1:8000/api/v2/books/?limit=6&offset=2  从第2条开始拿6条

在这里插入图片描述

从第二条数据之后开始指定获取数据的条数,使用了limit指定了获取6条,但是被max_limit=5给限制了,所以后端只能返回2条数据,如果不使用limit指定获取的数据条数,那么默认使用default_limit=2后端会返回2条数据。


分页器三:CursorPagination(游标分页)

自定义分页类

	from rest_framework.pagination import CursorPaginationclass CommonCursorPagination(CursorPagination):cursor_query_param = 'cursor'  # 按游标查询的查询条件,value值前端是不知道的,只能通过后台返回page_size = 2  # 每页显示多少条啊ordering = 'id'  # 排序规则,必须是表中的字段

视图类

	from .pagination import CommonCursorPaginationclass BookView(ViewSetMixin,ListAPIView):queryset = models.Book.objects.all()serializer_class = BookSerializer'注意分页方式一个接口只能使用一种,所以没有加中括号''另外使用游标分页的方式后就不能再其中使用排序了,因为其内部已经排好序了'pagination_class = CommonCursorPagination'并且只能选择上一页和下一页,不能指定跳转某一页,但是速度快,针对与特别大的数据量,游标分页有优势'

在这里插入图片描述


四、异常处理

REST framework提供了异常处理,我们可以自定义异常处理函数。主要是为了DRF在运行过程中发生的错误进行一个捕获,然后响应友好的提示信息给前端。

自定义异常处理函数:(定制一个异常处理统一的返回格式)

	from rest_framework.views import exception_handler,Response# drf的异常处理是exception_handler处理了,到那时没处理非drf的异常# 自己写个函数,处理drf异常和自己的异常,以后只要是出了异常都会走这里def Common_exception_handler(exc, context):# exc:异常信息# context:执行请求的视图、args、kwargs参数、request请求res = exception_handler(exc, context)# 在此处补充自定义的异常处理if res:  # 有值说明是drf的异常,如果没有值说明是自己的异常# data = {'detail': exc.detail}detail = res.data.get('detail') or "drf异常,请联系系统管理员"return Response({'code':666,'message':detail})else:return Response({'code':777,'message':f"系统异常,请联系管理{str(exc)}"})

在配置文件中声明自定义的异常处理

	REST_FRAMEWORK = {'EXCEPTION_HANDLER': 'appo1.exception.Common_exception_handler',}

如果未声明,会采用默认的方式,如下:

	REST_FRAMEWORK = {'EXCEPTION_HANDLER': 'rest_framework.views.exception_handler'}

视图类

	'''全局异常处理'''from rest_framework.exceptions import AuthenticationFailed,APIException'''Drf中无论在三大认证还是视图类中 方法执行报错包括主动抛异常都会执行一个函数excption_exception处理异常的函数只要出了异常APIView的dispatch中就可以捕获到 执行配置文件中的'EXCEPTION_HANDLER':'rest_framework.views.exception_handler''''class UserView(ViewSetMixin, APIView):def create(self, request):raise Exception('你出错了')# drf默认能处理自己的异常,它的异常都是继承自APIException的异常# raise AuthenticationFailed('认证失败')return Response({'code':100,'msg':'1111'})

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/226392.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux 线程概念

文章目录 前言线程的概念线程的操作操作的原理补充与说明 前言 ① 函数的具体说明被放在补充与说明部分 ② 只说些基础概念和函数使用 线程的概念 网络回答:Linux 线程是指在 Linux 操作系统中创建和管理的轻量级执行单元。线程是进程的一部分,与进程…

【电子通识】开关的种类

开关在我们日常生活与工作中使用较多。开关有无数种形式,种类繁多。从微小的按钮到巨大的控制器,功能多种多样。这种多样性受到机械或电气操作、手动或电子控制等因素的影响,并且与个人在设计美学和用户界面方面的偏好也有关。 电子开关采用 …

LabVIEW利用视觉引导机开发器人精准抓取

LabVIEW利用视觉引导机开发器人精准抓取 本项目利用单目视觉技术指导多关节机器人精确抓取三维物体的技术。通过改进传统的相机标定方法,结合LabVIEW平台的Vision Development和Vision Builder forAutomated Inspection组件,优化了摄像系统的标定过程&a…

低代码平台在金融银行中的应用场景

随着数字化转型的推进,商业银行越来越重视技术在业务发展中的作用。在这个背景下,白码低代码平台作为一种新型的开发方式,正逐渐受到广大商业银行的关注和应用。白码低代码平台能够快速构建各类应用程序,提高开发效率,…

概率论相关题型

文章目录 概率论的基本概念放杯子问题条件概率与重要公式的结合独立的运用 随机变量以及分布离散随机变量的分布函数特点连续随机变量的分布函数在某一点的值为0正态分布标准化随机变量函数的分布 多维随机变量以及分布条件概率max 与 min 函数的相关计算二维随机变量二维随机变…

<JavaEE> TCP 的通信机制(五) -- 延时应答、捎带应答、面向字节流

目录 TCP的通信机制的核心特性 七、延时应答 1)什么是延时应答? 2)延时应答的作用 八、捎带应答 1)什么是捎带应答? 2)捎带应答的作用 九、面向字节流 1)沾包问题 2)“沾包…

NXP实战笔记(三):S32K3xx基于RTD-SDK在S32DS上配置WDT配置

目录 1、WDT概述 2、SWT配置 2.1、超时时间,复位方式的配置 2.2、中断形式 1、WDT概述 SWT 编程模型只允许 32 位(字)访问。 以下任何尝试访问都是无效的: •非32位访问 •写入只读寄存器 •启用SWT时,将不正确的值写入SR…

Mongodb基础介绍与应用场景

NoSql 解决方案第二种 Mongodb MongoDB 是一款开源 高性能 无模式的文档型数据库 当然 它是NoSql数据库中的一种 是最像关系型数据库的 非关系型数据库 首先 最需要注意的是 无模式的文档型数据库 这个需要后面我们看到它的数据才能明白 其次是 最像关系型数据库的非关系型数据…

基于采样的自动驾驶规划算法 - PRM,RRT,RRT*,CL-RRT

本文将讲解PRM,RRT,RRT*自动驾驶规划算法原理,不正之处望读者指正 0 前言 机器人运动规划的基本任务:从开始位置到目标位置的运动 (1)如何躲避构型空间出现的障碍物 (2)如何满足机器…

小型企业成为网络犯罪分子获取数据的目标

在过去十年的大部分时间里,网络犯罪的巨额资金来自针对大型组织的勒索软件攻击。这种威胁仍然存在。但犯罪分子可能会将注意力转向中小企业 (SMB)。这对消费者的影响将是巨大的。 将软件即服务 (SaaS) 技术用于核心业务功能继续将中小企业整合到全球供应链中。由于…

【C语言】数据结构——排序(一)

💗个人主页💗 ⭐个人专栏——数据结构学习⭐ 💫点击关注🤩一起学习C语言💯💫 目录 导读:数组打印与交换1. 插入排序1.1 直接插入排序1.1.1 基本思想1.1.2 实现代码1.1.3 图解 1.2 希尔排序1.2.1…

C语言之进制转换

C语言之进制转换 一、引言二、十进制与二进制、八进制、十六进制三、二进制与八进制、十六进制四、八进制与十六进制 一、引言 在C语言中,经常使用的整数的进制有十进制、二进制、十六进制(在C语言中以0x或0X为前缀)、八进制(在C…

如何手动升级Chrome插件/Chrome扩展程序?

Chrome 浏览器的插件(也称为扩展)通常会自动更新到最新版本。这是因为 Chrome 会定期检查并下载来自 Chrome 网上应用店的扩展更新。然而,如果你需要手动更新扩展,可以按照以下步骤操作: 打开 Chrome 浏览器。点击浏览…

Three.js基础入门介绍——Three.js学习三【借助控制器操作相机】

在Three.js基础入门介绍——Three.js学习二【极简入门】中介绍了如何搭建Three.js开发环境并实现一个包含旋转立方体的场景示例,以此为前提,本篇将引进一个控制器的概念并使用”轨道控制器”(OrbitControls)来达到从不同方向展示场…

Flink项目实战篇 基于Flink的城市交通监控平台(上)

系列文章目录 Flink项目实战篇 基于Flink的城市交通监控平台(上) Flink项目实战篇 基于Flink的城市交通监控平台(下) 文章目录 系列文章目录1. 项目整体介绍1.1 项目架构1.2 项目数据流1.3 项目主要模块 2. 项目数据字典2.1 卡口…

车路协同中 CUDA 鱼眼相机矫正、检测、追踪

在车路协同中,鱼眼一般用来补充杆件下方的盲区,需要实现目标检测、追踪、定位。在目标追踪任务中,通常的球机或者枪机方案,无法避免人群遮挡的问题,从而导致较高的ID Swich,造成追踪不稳定。但是鱼眼相机的顶视角安装方式,天然缓解了遮挡的问题,从而实现杆件下方的盲区…

51单片机(STC8)-- GPIO输入输出

文章目录 I/O口相关寄存器端口数据寄存器端口模式配置寄存器(PxM0,PxM1)端口上拉电阻控制寄存器(PxPU)关于I/O的注意事项 配置I/O口I/O设置demoI/O端口模式LED控制(I/O输出)按键检测(I/O输入) S…

SpringBoot多线程与任务调度总结

一、前言 多线程与任务调度是java开发中必须掌握的技能,在springBoot的开发中,多线程和任务调度变得越来越简单。实现方式可以通过实现ApplicationRunner接口,重新run的方法实现多线程。任务调度则可以使用Scheduled注解 二、使用示例 Slf…

vue3中使用defineComponent封装hook实现模板复用

文章目录 一、前言二、useTemplate 实现三、最后 一、前言 最近在做 Vue3 项目的时候&#xff0c;在思考一个小问题&#xff0c;其实是每个人都做过的一个场景&#xff0c;很简单&#xff0c;看下方代码 <template><div><div v-for"item in data" :…

Eclipse安装Jrebel eclipse免重启加载项目

每次修改JAVA文件都需要重新启动项目&#xff0c;加载时间太长&#xff0c;eclipse安装jrebel控件,避免重启项目节省时间。 1、Help->Eclipse Marketplace 2、搜索jrebel 3、Help->jrebel->Configuration 配置jrebel 4、激活jrebel 5、在红色框中填入 http://jrebel…