rest framework之过滤组件
一、普通过滤
(一)get_queryset
get_queryset方法是GenericAPIView提供的一个方法,旨在返回queryset数据集,而过滤就是要在这个方法返回数据集之前对数据进行筛选,然后返回筛选后的数据即可,那么也就是要求需要重写这个方法:
def get_queryset(self):
"""
Get the list of items for this view.
This must be an iterable, and may be a queryset.
Defaults to using `self.queryset`. This method should always be used rather than accessing `self.queryset`
directly, as `self.queryset` gets evaluated only once, and those results
are cached for all subsequent requests. You may want to override this if you need to provide different
querysets depending on the incoming request. (Eg. return a list of items that is specific to the user)
"""
assert self.queryset is not None, (
"'%s' should either include a `queryset` attribute, "
"or override the `get_queryset()` method."
% self.__class__.__name__
) queryset = self.queryset
if isinstance(queryset, QuerySet):
# Ensure queryset is re-evaluated on each request.
queryset = queryset.all()
return queryset
源码中的这个方法会从视图配置中获得queryset,然后判断是否属于QuerySet类型,如果属于,就会返回这个queryset。
(二)实例
- 请求API
class BookView(GenericViewSet):
serializer_class = BookModelSerializer
def get_queryset(self):
queryset = models.Book.objects.all()
title = self.request.query_params.get('title',None)
if title is not None:
queryset = models.Book.objects.filter(title=title).all()
return queryset
def list(self,request,*args,**kwargs):
queryset= self.get_queryset()
bs=self.get_serializer(queryset,many=True)
return Response(bs.data)
- 路由配置:
注意如果需要过滤,url中需要加入base_name属性,并且base_name的值就是过滤参数的名称:
router=routers.DefaultRouter()
router.register('books',views.BookView,base_name="title")
- 生成的路由:
^books/$ [name='title-list']
^books\.(?P<format>[a-z0-9]+)/?$ [name='title-list']
- 请求地址:
http://127.0.0.1:8020/books/?title=语文
二、django-filter过滤
(一)django-filter的简单使用
1、安装django-filter
pip install django-filter
2、注册django-filter
INSTALLED_APPS = [
'django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
'app01.apps.App01Config',
'rest_framework',
'django_filters', #注册django-filter
]
3、设置通用过滤后端
- 全局配置
REST_FRAMEWORK = {
'DEFAULT_FILTER_BACKENDS': ('django_filters.rest_framework.DjangoFilterBackend',)
}
- 局部视图设置
import django_filters.rest_framework
class BookView(GenericViewSet):
queryset = models.Book.objects.all()
serializer_class = BookModelSerializer
filter_backends = (django_filters.rest_framework.DjangoFilterBackend,)
4、指定筛选字段
如果需求都是些简单类型的筛选,那么可以在view或viewSet里面设置一个filter_fields属性,列出所有依靠筛选的字段集合。
import django_filters.rest_framework
from rest_framework.viewsets import ModelViewSet class BookView(ModelViewSet): ...
filter_fields = ('title',) #列出搜索字段
...
5、进行访问
- 路由配置
from rest_framework import routers
router=routers.DefaultRouter()
router.register('books',views.BookView) #无需base_name参数
urlpatterns = [
re_path('',include(router.urls)),
]
- 生成的路由
^books/$ [name='book-list']
^books\.(?P<format>[a-z0-9]+)/?$ [name='book-list']
^books/(?P<pk>[^/.]+)/$ [name='book-detail']
^books/(?P<pk>[^/.]+)\.(?P<format>[a-z0-9]+)/?$ [name='book-detail']
- 进行访问
http://127.0.0.1:8020/books/?title=语文
(二)FilterSet的简单使用
上面所搜索的字段是一对一的关系,没有涉及到外键以及多对多字段,如果有外键或者多对多关系,可以在filter_fields中使用‘__’进行跨越:
class BookView(ModelViewSet):
queryset = models.Book.objects.all()
serializer_class = BookModelSerializer
filter_backends = (django_filters.rest_framework.DjangoFilterBackend,)
# 列出搜索字段,其中publish__name为ForeignKey字段,authors__name为ManyToMany字段
filter_fields = ('title','publish__name','authors__name')
class Book(models.Model):
title=models.CharField(max_length=32)
price=models.IntegerField()
pub_date=models.DateField(null=True,blank=True)
publish=models.ForeignKey("Publish",on_delete=models.CASCADE)
authors=models.ManyToManyField("Author")
def __str__(self):
return self.title
models
此时请求地址:
#含外键
http://127.0.0.1:8020/books/?title=语文&publish__name=北京出版社 #含多对多
http://127.0.0.1:8020/books/?title=语文&publish__name=北京出版社&authors__name=张三
但这样外键和多对多将Django的双下划线约定作为API的一部分暴露出来。如果想显式地命名过滤器参数,可以显式地将它包含在FilterSet类中:
1、定义FilterSet类
import django_filters
from app01 import models class BookFilter(django_filters.rest_framework.FilterSet):
publish = django_filters.CharFilter(field_name="publish__name")
authors = django_filters.CharFilter(field_name="authors__name")
class Meta:
model = models.Book
fields = ['title', 'publish', 'authors']
2、在视图中使用定义的FilterSet类
class BookView(ModelViewSet):
queryset = models.Book.objects.all()
serializer_class = BookModelSerializer
filter_backends = (django_filters.rest_framework.DjangoFilterBackend,)
filter_class = BookFilter #使用定义的FilterSet类
此时请求地址:
http://127.0.0.1:8020/books/?title=语文&publish=北京出版社&authors=张三
3、总结
FilterSet可以实现较为复杂的过滤功能,如果是普通字段可以使用filter_fields 添加字段进行过滤,如果有外键、多对多以及范围(日期范围、价格范围)使用FilterSet.
三、其它过滤
(一)搜索过滤(SearchFilter)
支持基于简单查询参数的搜索,并且基于Django admin的搜索功能。
- 设置过滤后端
from rest_framework import filters class BookView(ModelViewSet): ...
filter_backends = (filters.SearchFilter,)
...
- 设置查询字段
from rest_framework import filters
class BookView(ModelViewSet):
queryset = models.Book.objects.all()
serializer_class = BookModelSerializer
# filter_backends = (django_filters.rest_framework.DjangoFilterBackend,)
filter_backends = (filters.SearchFilter,)
#凡是以下字段内容都会被搜到
search_fields = ('title','publish__name','authors__name')
当然可以通过在search_fields 前面添加各种字符来限制搜索行为:
'^' 以指定内容开始. '=' 完全匹配 '@' 全文搜索(目前只支持Django的MySQL后端) '$' 正则搜索
例如:
search_fields = ('=title',)
- 客户端访问
#将北京出版社的内容全部搜索出来
http://127.0.0.1:8020/books/?search=北京出版社
#将张三相关内容搜索出来
http://127.0.0.1:8020/books/?search=张三
通过关键词search进行搜索,可以搜索后端search_fields提供字段的内容。
SearchFilters类的默认搜索关键字为search,可以通过SEARCH_PARAM进行覆盖设置
class BookView(ModelViewSet):
...
SEARCH_PARAM = 'ss'
...
(二)排序筛选(OrderingFilter)
- 设置排序筛选后端
from rest_framework import filters class BookView(ModelViewSet): ...
filter_backends = (filters.OrderingFilter,)
...
- 指定支持排序的字段(ordering_fields )
from rest_framework import filters class BookView(ModelViewSet):
...
filter_backends = (OrderingFilter,)
ordering_fields = ('title', 'id') #指定支持排序的字段
...
- 客户端访问
通过关键字ordering请求返回的数据按照什么排序
#按照title排序
http://127.0.0.1:8020/books/?ordering=title #按照id排序
http://127.0.0.1:8020/books/?ordering=id
#按照id逆序
http://127.0.0.1:8020/books/?ordering=-id #按照多个字段排序
http://127.0.0.1:8020/books/?ordering=id,title
- 默认排序
如果需要在返回的数据中已经排好序了,可以在视图中使用ordering参数:
from rest_framework import filters
class BookView(ModelViewSet):
queryset = models.Book.objects.all()
serializer_class = BookModelSerializer
filter_backends = (filters.OrderingFilter,)
ordering_fields = ('title', 'id')
ordering=('-id',) #默认排序
客户端访问:
http://127.0.0.1:8020/books/
返回的是已经按照id逆序排列的数据。
参考文档:https://q1mi.github.io/Django-REST-framework-documentation/api-guide/filtering_zh/#filtering
rest framework之过滤组件的更多相关文章
- DRF框架(八)——drf-jwt手动签发与校验、搜索过滤组件、排序过滤组件、基础分页组件
自定义drf-jwt手动签发和校验 签发token源码入口 前提:给一个局部禁用了所有 认证与权限 的视图类发送用户信息得到token,其实就是登录接口,不然进不了登录页面 获取提交的username ...
- REST framework之分页组件
REST framework之分页组件 一 简单分页 查看第n页,每页显示n条 from rest_framework.pagination import PageNumberPagination # ...
- drf框架 - 过滤组件 | 分页组件 | 过滤器插件
drf框架 接口过滤条件 群查接口各种筛选组件数据准备 models.py class Car(models.Model): name = models.CharField(max_length=16 ...
- DRF内置过滤组件与排序组件结合使用
DRF内置过滤组件Filtering DRF提供了内置过滤组件Filtering,可以结合url路径的改变获取想要的数据,当然用户不可能在url访问路径中自己设置过滤条件,肯定是后端开发人员将前端页面 ...
- rest framework之路由组件
一.路由组件的使用 1.使用实例 在视图中继承GenericViewSet类来完成功能时,需要自己对路由的写法有所改变,需要在as_view中传入actions字典参数: re_path('books ...
- restful framework之序列化组件
一.Django自带序列化组件 from django.core import serializers def test(request): book_list = Book.objects.all( ...
- restful framework之权限组件
一.权限简介 只用超级用户才能访问指定的数据,普通用户不能访问,所以就要有权限组件对其限制 二.局部使用 from rest_framework.permissions import BasePerm ...
- restful framework之认证组件
一.认证介绍 只有认证通过的用户才能访问指定的url地址,比如:查询课程信息,需要登录之后才能查看,没有登录,就不能查看,这时候需要用到认证组件 二.局部使用 (1)models层: class Us ...
- RESTful源码笔记之RESTful Framework的基本组件
快速实例 Quickstart 序列化 创建一个序列化类 简单使用 开发我们的Web API的第一件事是为我们的Web API提供一种将代码片段实例序列化和反序列化为诸如json之类的表示形式的方式. ...
随机推荐
- Linux Shell脚本在service模式下的环境变量
普通Shell脚本在终端下运行时,环境变量来自于init/profile/.bash_profile等,就是在当前终端下可以得到的环境,比如PATH.在service模式下,脚本运行在受限的环境中:比 ...
- matlab 代码分析
在command window中输入 >> profile on>> profile clear>> profile viewer 就会出现如下窗口 在将头所指向的 ...
- pipeline语法之environment,dir(),deleteDir()方法,readJSON,writeJSON
一 environment指令指定一系列键值对,这些对值将被定义为所有步骤的环境变量或阶段特定步骤 environment{…}, 大括号里面写一些键值对,也就是定义一些变量并赋值,这些变量就是环境变 ...
- &与&&、|与||的区别
&和&& 相同之处: &和&&都表示:符号两端必须同时为真,最后的结果才为真:其中一端为假,则最后的结果为假 不同之处: &:左端为假,还需要继 ...
- 理解webpack中的process.env.NODE_ENV
参考资料 一. process 要理解 process.env.NODE_ENV 就必须要了解 process,process 是 node 的全局变量,并且 process 有 env 这个属性,但 ...
- BZOJ 3207: 花神的嘲讽计划Ⅰ(莫队+哈希)
传送门 解题思路 刚开始写了个莫队+哈希+\(map\)的\(O(n\sqrt(n)log(n)\)的辣鸡做法,\(T\)飞了.后来看了看别人博客发现其实并不用拿\(map\)当桶存那些哈希值.因为只 ...
- vue2 — vuex状态管理
一.为什么使用vuex : https://www.cnblogs.com/goloving/p/9080005.html vuex的功能 和 localstorage 的作用是一样,把数据在一个所有 ...
- 前端每日实战:58# 视频演示如何用纯 CSS 创作一只卡通鹦鹉
效果预览 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/vrRmWy 可交互视频 此视频是可 ...
- HDU 6625 three arrays 求两个序列异或最小值的排列(一个可以推广的正解
目录 题意: 解析 原题描述 字典树动态求Mex @(hdu 6625求两个序列异或最小值的排列) 题意: \(T(100)\)组,每组两个长度为\(n(100000)\)的排列,你可以将\(a[]\ ...
- mysql left join 用法说明
left join中关于where和on条件的几个知识点: 1.多表left join是会生成一张临时表,并返回给用户 2.where条件是针对最后生成的这张临时表进行过滤,过滤掉不符合where条件 ...