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

相关文章

一文详解Java Stream的sorted自定义排序

《一文详解JavaStream的sorted自定义排序》Javastream中的sorted方法是用于对流中的元素进行排序的方法,它可以接受一个comparator参数,用于指定排序规则,sorte... 目录一、sorted 操作的基础原理二、自定义排序的实现方式1. Comparator 接口的 Lam

HTML5 搜索框Search Box详解

《HTML5搜索框SearchBox详解》HTML5的搜索框是一个强大的工具,能够有效提升用户体验,通过结合自动补全功能和适当的样式,可以创建出既美观又实用的搜索界面,这篇文章给大家介绍HTML5... html5 搜索框(Search Box)详解搜索框是一个用于输入查询内容的控件,通常用于网站或应用程

Java List排序实例代码详解

《JavaList排序实例代码详解》:本文主要介绍JavaList排序的相关资料,Java排序方法包括自然排序、自定义排序、Lambda简化及多条件排序,实现灵活且代码简洁,文中通过代码介绍的... 目录一、自然排序二、自定义排序规则三、使用 Lambda 表达式简化 Comparator四、多条件排序五、

JAVA数组中五种常见排序方法整理汇总

《JAVA数组中五种常见排序方法整理汇总》本文给大家分享五种常用的Java数组排序方法整理,每种方法结合示例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录前言:法一:Arrays.sort()法二:冒泡排序法三:选择排序法四:反转排序法五:直接插入排序前言:几种常用的Java数组排序

对Django中时区的解读

《对Django中时区的解读》:本文主要介绍对Django中时区的解读方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录背景前端数据库中存储接口返回AI的解释问题:这样设置的作用答案获取当前时间(自动带时区)转换为北京时间显示总结背景设置时区为北京时间 TIM

Springboot实现推荐系统的协同过滤算法

《Springboot实现推荐系统的协同过滤算法》协同过滤算法是一种在推荐系统中广泛使用的算法,用于预测用户对物品(如商品、电影、音乐等)的偏好,从而实现个性化推荐,下面给大家介绍Springboot... 目录前言基本原理 算法分类 计算方法应用场景 代码实现 前言协同过滤算法(Collaborativ

Django之定时任务django-crontab的实现

《Django之定时任务django-crontab的实现》Django可以使用第三方库如django-crontab来实现定时任务的调度,本文主要介绍了Django之定时任务django-cront... 目录crontab安装django-crontab注册应用定时时间格式定时时间示例设置定时任务@符号

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

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

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

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

Django序列化中SerializerMethodField的使用详解

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