Django REST framwork-07-过滤、搜索和排序

2024-03-23 20:20

本文主要是介绍Django REST framwork-07-过滤、搜索和排序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、简介

REST框架的通用列表视图的默认行为是返回模型管理器的整个查询集。

筛选子类的任何视图的查询集的最简单方法是覆盖该GenericAPIView 的 .get_queryset() 方法。

基于 URL 传递的参数过滤

  1. URL
    考虑如下 URL 配置条目:
path('servers-base-url-args/<manufacturer>', views.ServersListView.as_view()),
  1. URL 请求
    http://www.qfsite.com/api/servers-base-url-args/del/
  2. 视图
    URL 传递的参数会在 self.kwargs 或者 self.args 中
class ServersListView(generics.ListAPIView):serializer_class = ServersSerializerdef get_queryset(self):"""默认返回所有的服务器,这里使用了过滤,只返回某一个厂商的。"""manufacturer = self.kwargs['manufacturer']return Servers.objects.filter(manufacturer=manufacturer)

基于 GET 方法中的参数过滤

  1. URL
    考虑如下 URL 配置条目:
path('servers-base-get-args/', ServersListView.as_view()),
  1. URL 请求
    http://www.qfsite.com/api/servers-base-get-args/?manufacturer=del
  2. 视图
    GET 方法传参是在 self.request.query_params 中,这是一个字典。
class ServersListView(generics.ListAPIView):serializer_class = ServersSerializerdef get_queryset(self):"""默认返回所有的服务器,这里使用了过滤,只返回某一个厂商的。"""queryset = Servers.objects.all()manufacturer = self.request.query_params.get('manufacturer', None)if manufacturer:queryset = queryset.filter(manufacturer=manufacturer)return queryset

通用过滤器(重点掌握)

REST框架还包括对通用过滤后端的支​​持,允许您轻松构建复杂的搜索和过滤器。

通用过滤器还可以在可浏览的API和管理API中将自身呈现为HTML控件。

在这里插入图片描述
在这里插入图片描述

配置过滤器后端 django-filters

DjangoFilterBackend
该 django-filter 库包含一个 DjangoFilterBackend 类,该类支持 REST 框架的高度可定制的字段过滤。

要使用 DjangoFilterBackend,请先安装 django-filter。然后添加 django_filters 到 Django的 INSTALLED_APPS

pip3 install django-filter
  1. 注册 APP
    在 settings.py 中设置
INSTALLED_APPS = [...'django_filters',
]

注意: 中间的是下划线 _

2.1 为全局设置
您现在应该将过滤器后端添加到您的设置:

在 settings.py 中设置

REST_FRAMEWORK = {'DEFAULT_FILTER_BACKENDS': ['django_filters.rest_framework.DjangoFilterBackend']
}

2.2 或者为具体某个视图设置
或者将过滤器后端添加到单个View或ViewSet。

view 或者 ViewSet 必须是基于 GenericAPIView 的类视图

from django_filters.rest_framework import DjangoFilterBackend
class UserListView(generics.ListAPIView):...filter_backends = [DjangoFilterBackend]
  1. 如何使用
    3.1 视图
    过滤字段的值必须是 等于的关系
    只需要在视图或视图集上设置 filterset_fields 属性,并列出要过滤的字段集
from django_filters.rest_framework import DjangoFilterBackendclass ServersListView(generics.ListAPIView):queryset = Servers.objects.all()serializer_class = ServersSerializerfilter_backends = [DjangoFilterBackend]filterset_fields = ['hostname', 'manufacturer']

3.2 URL 配置

urlpatterns = [path("servers-filter/", views.ServersFilterView.as_view()),
]

3.3 URL 请求
这将接受下面的请求:

http://www.qfedu.com/api/servers-filter/?hostname=db.server.yy.com&manufacturer=
3.4 效果图
填写过滤条件
在这里插入图片描述

过滤结果:
在这里插入图片描述

可以发现 django-filter 默认的过滤器支持的只是精确匹配。
假如希望进行模糊匹配,比如希望查找到所有包含 db.server 的主机名等。
这样的需求可以采用自定义过滤器的的方式。

自定义过滤类

利用 django-filter 自定义过滤类,对于更高级的过滤要求,您可以自己编写一个过滤器,这个过滤器很想 Django 中的 From 类 ,之后在指定视图中使用 filterset_class=自定义类名 来使用这个类。

1. 编写自定义过滤器的类

这个过滤器的目标是: 查找到创建时间在某个范围的服务器。

from django_filters import rest_framework as filters
from users.models import Serverclass ServersFilter(filters.FilterSet):"""服务器的过滤类"""# 可以自定义筛选器属性名,如 create_at_mincreate_at_min = filters.DateTimeFilter(field_name='create_at', lookup_expr='gte',label="创建时间 大于等于")create_at_max = filters.DateTimeFilter(field_name='create_at', lookup_expr='lte', label="创建时间 小于等于")class Meta:model = Serverfields = ['create_at_min', 'create_at_max']

核心参数

  • field_name: 筛选依据的模型字段的名称。如果未提供此参数,则会默认FilterSet类上的筛选器属性名称。字段名可以通过将相关部分与ORM查找分隔符(__)连接来遍历关系。例如 manufacturer__name
  • lookup_eper: 应在筛选器调用中执行的字段查找。默认为精确匹配。如果表达式部分由ORM查找分隔符(__)连接,则 lookup_expr 可以包含更多的查找转换规则。例如,通过日期时间的年部分 year__gt 来过滤日期时间。

label 属性的值会显示到页面提示上。

  1. 在视图中使用
from django_filters.rest_framework import DjangoFilterBackend
from .filters import ServersFilter
class ServersFilterViewSet(viewsets.ReadOnlyModelViewSet):queryset = Server.objects.all()serializer_class = ServerSerializerfilter_backends = [DjangoFilterBackend]filterset_class = ServersFilter
  1. URL 配置
    由于上面的视图是采用视图集(ViewSet)的方式,所以可以使用路由注册的方式。
router.register(r'servers-filter', views.ServersFilterViewSet)
  1. 效果图
    在这里插入图片描述

  2. 支持字段值的模糊匹配
    对主机名 hostname 进行模糊匹配,过滤器需进行如下配置
    模糊匹配相当于 MySql 中的 like.
    这里使用 icontains, i 是忽略大小写的。

class ServersFilter(filters.FilterSet):"""服务器的过滤类"""create_at = filters.DateFromToRangeFilter(label="创建时间范围搜索after-before")hostname = filters.CharFilter(field_name='hostname', lookup_expr='icontains')class Meta:model = Serverfields = ['create_at', 'hostname']

注意: Model 中的字段是 models.DateTimeField (年-月-日 时:分:秒).
在这里仍然可以使用 filters.DateFromToRangeFilter (年-月-日) 的方式进行过滤。

在这里插入图片描述

关键字参数 method

一个可选参数,告诉筛选器如何处理查询集。它可以接受FilterSet上的方法名称或者任意位置编写的函数。方法或函数接收一个QuerySet queryset、要筛选的模型字段的名称 name以及要筛选的值 value。它应该返回一个经过筛选的Queryset。

请注意,该值由Filter.field验证,因此不需要原始值转换和空值检查。

参考官方示例如下:

class F(FilterSet):"""根据图书是否已出版筛选图书"""published = BooleanFilter(field_name='published_on', method='filter_published')def filter_published(self, queryset, name, value):# 构造完整的查找表达式。lookup = '__'.join([name, 'isnull'])return queryset.filter(**{lookup: False})# 或者, 对于 lookup 你可以使用硬编码。 比如,# return queryset.filter(published_on__isnull=False)class Meta:model = Bookfields = ['published']# 也可以在类作用域之外定义一个函数
def filter_not_empty(queryset, name, value):lookup = '__'.join([name, 'isnull'])return queryset.filter(**{lookup: False})class F(FilterSet):"""根据图书是否已出版筛选图书"""published = BooleanFilter(field_name='published_on', method=filter_not_empty)class Meta:model = Bookfields = ['published']

更多
点我官方文档-过滤参考

支持跨表字段的匹配

如下示例中使用的 model:

class Servers(models.Model):"""服务器"""ENV = (("0", "测试"),("1", "生产"),("2", "开发"),("3", "预生产"),)project = models.ForeignKey("projects.Projects", related_name="servers", on_delete=models.CASCADE, verbose_name="所属项目", help_text="服务器属于哪个项目")env = models.CharField(max_length=1, choices= ENV,default="1", verbose_name="环境", help_text="哪个环境的服务器")host_name = models.CharField(max_length=128, unique=True, db_index=True, verbose_name="主机名", help_text="主机名")class Meta:"""元数据"""verbose_name = "服务器表"verbose_name_plural = verbose_nameclass Projects(models.Model):"""项目表"""name = models.CharField(max_length=30, verbose_name="项目名称", help_text="项目名称")class Meta:"""定义数据库中实际的表名称"""verbose_name = "项目表"verbose_name_plural = verbose_name

实现关系字段的过滤,可以使用下面的示例:

class ServersFilter(filters.FilterSet):"""服务器的过滤类"""project = filters.CharFilter(field_name='project__name', lookup_expr='icontains', label="所属项目")class Meta:model = Serversfields = ["env", "local_ip", "project"]

field_name 的值 project__name ,表示了关系字段的查询。
fields 中 project 是过滤器的属性名称,也是前端发送请求时候需要传递的 GET 请求参数名称。
lookup_expr 的 icontains 表示 包含的意思。
查询URL:
http://localhost:8000/servers/?project=项目名

SearchFilter 搜索

搜索使用的是 Django rest framework 自己的 SearchFilter

同样在浏览器 API 中会展现出一个搜索控件,可以方便前端人员、测试人员的测试。

视图

from rest_framework import filters
class ServersSearchViewSet(viewsets.ReadOnlyModelViewSet):queryset = Server.objects.all()serializer_class = ServerSerializerfilter_backends = [filters.SearchFilter]search_fields = ['hostname', 'manufacturer']

URL 配置

router.register(r'servers-search', views.ServersSearchViewSet)

在这里插入图片描述

可以支持基于双下划线的夸表搜索

search_fields = ['hostname', 'manufacturer', ' 'disk__capacity']

这样会增加搜索 硬盘容量在指定范围的 服务器
在这里插入图片描述

支持组合搜索
搜索主机名还有 db 字符并且硬盘容量是 200 的服务器

在这里插入图片描述

搜索的逻辑关系
默认搜索的条件是模糊匹配的,就是包含的意思。

还支持以下逻辑关系

  • '^' 匹配开头。
  • '=' 完全匹配。
  • '@' 全文搜索。(目前只支持Django的MySQL后端。)
  • '$' 正则表达式搜索。

示例

search_fields = ['$hostname']

在这里插入图片描述

OrderingFilter 排序

class ServersSearchViewSet(viewsets.ReadOnlyModelViewSet):queryset = Server.objects.all()serializer_class = ServerSerializerfilter_backends = [filters.SearchFilter, filters.OrderingFilter]search_fields = ['$hostname', 'manufacturer', 'disk__capacity']ordering_fields = ["create_at", "disk__capacity"]ordering = ["-create_at"]  # 默认排序字段

少张图

这篇关于Django REST framwork-07-过滤、搜索和排序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Mybatis 传参与排序模糊查询功能实现

《Mybatis传参与排序模糊查询功能实现》:本文主要介绍Mybatis传参与排序模糊查询功能实现,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、#{ }和${ }传参的区别二、排序三、like查询四、数据库连接池五、mysql 开发企业规范一、#{ }和${ }传参的

SpringIntegration消息路由之Router的条件路由与过滤功能

《SpringIntegration消息路由之Router的条件路由与过滤功能》本文详细介绍了Router的基础概念、条件路由实现、基于消息头的路由、动态路由与路由表、消息过滤与选择性路由以及错误处理... 目录引言一、Router基础概念二、条件路由实现三、基于消息头的路由四、动态路由与路由表五、消息过滤

Django序列化中SerializerMethodField的使用详解

《Django序列化中SerializerMethodField的使用详解》:本文主要介绍Django序列化中SerializerMethodField的使用,具有很好的参考价值,希望对大家有所帮... 目录SerializerMethodField的基本概念使用SerializerMethodField的

java streamfilter list 过滤的实现

《javastreamfilterlist过滤的实现》JavaStreamAPI中的filter方法是过滤List集合中元素的一个强大工具,可以轻松地根据自定义条件筛选出符合要求的元素,本文就来... 目录1. 创建一个示例List2. 使用Stream的filter方法进行过滤3. 自定义过滤条件1. 定

Redis如何实现刷票过滤

《Redis如何实现刷票过滤》:本文主要介绍Redis如何实现刷票过滤问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录引言一、概述二、技术选型三、搭建开发环境四、使用Redis存储数据四、使用SpringBoot开发应用五、 实现同一IP每天刷票不得超过次数六

C++快速排序超详细讲解

《C++快速排序超详细讲解》快速排序是一种高效的排序算法,通过分治法将数组划分为两部分,递归排序,直到整个数组有序,通过代码解析和示例,详细解释了快速排序的工作原理和实现过程,需要的朋友可以参考下... 目录一、快速排序原理二、快速排序标准代码三、代码解析四、使用while循环的快速排序1.代码代码1.由快

Python使用DeepSeek进行联网搜索功能详解

《Python使用DeepSeek进行联网搜索功能详解》Python作为一种非常流行的编程语言,结合DeepSeek这一高性能的深度学习工具包,可以方便地处理各种深度学习任务,本文将介绍一下如何使用P... 目录一、环境准备与依赖安装二、DeepSeek简介三、联网搜索与数据集准备四、实践示例:图像分类1.

Spring排序机制之接口与注解的使用方法

《Spring排序机制之接口与注解的使用方法》本文介绍了Spring中多种排序机制,包括Ordered接口、PriorityOrdered接口、@Order注解和@Priority注解,提供了详细示例... 目录一、Spring 排序的需求场景二、Spring 中的排序机制1、Ordered 接口2、Pri

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常

Python中lambda排序的六种方法

《Python中lambda排序的六种方法》本文主要介绍了Python中使用lambda函数进行排序的六种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们... 目录1.对单个变量进行排序2. 对多个变量进行排序3. 降序排列4. 单独降序1.对单个变量进行排序