目录
filter过滤类源码分析
我们从视图类中群查接口入口,去看看它内部是怎么实现过滤的
首先我们进入ListAPIView类中
实现群查功能的是它的第一个父类mixins中调用的,我们进入它的第一个父类
我们看到它的群查接口有一个filter_queryset
方法,此时一定要清楚属性的查找顺序,
此时的self是指的视图类,如果视图类中没有这个方法,那就去它的父类中去找,那我这里就直接告诉你它是在GenericAPIView通用视图类中
''' 源码 ''' def filter_queryset(self, queryset): """ Given a queryset, filter it with whichever filter backend is in use. You are unlikely to want to override this method, although you may need to call it either from a list view, or from a custom `get_object` method if you want to apply the configured filtering backend to the default queryset. """ for backend in list(self.filter_backends): queryset = backend().filter_queryset(self.request, queryset, self) return queryset
根据以上代码剖析,我们知道了循环了一个配置属性,我们往上找找看有么有这个属性
在GenericAPIView通用视图类中我们发现了,filter_backend是在api_settings中配置的,所以我们在去到api_settings中看看他配置的是啥?
基于以上的验证,django自己配置的filter是空的列表,也就是没有默认不去过滤,需要我们自己去配置,而且在rest_framework内部一定实现了方法来启动我们配置的filter_backend
所以我们现在来看一下再rest_framework中到底是哪里实现了filter配置启动的方法
我们就直接从api_settings为入口找到rest_framework的配置文件包
好了到这里,我们就直接引出下面我们要说的两个部分的内容,排序、 搜索
排序组件 OrderingFilter
源码分析
根据以上的推理我们已经找到了filter配置的入口了,那紧接着我们就来看一下在上面的OrderingFilter类到底做了哪些事情
我们看到在OrderingFilter类 和 SearchFilter类中都继承了BaseFilterBackend,我们看下BaseFilterBackend类
看到父类中说明,在它的子类中,必须重写这个filter_queryset方法才可以用,我们在来看看他的子类OreringFilter**
将上述方法的结果返回给get_ordering,get_ordering将结果返回给filter_queryset,然后执行queryset.order(字段名)。queryset就是我们自已在view视图类中定义好的模型类。
使用说明
- 在使用的视图类中导入排序类
OrderingFilter
- 配置
filter_backends
- 配置参与排序的字段
ordering_fields
- (了解)可选配置:
ordering_param
规定接口中使用过滤的关键字
from . import models from rest_framework.generics import ListAPIView # 在使用的视图类中导入排序类 `OrderingFilter from rest_framework.filters import OrderingFilter class FreeCourseListAPIView(ListAPIView): queryset = models.Course.objects.filter(is_delete=False, is_show=True, ).order_by('-orders').all() serializer_class = serializers.FreeCourseModelSerializer # 配置 排序的过滤类,相当于是局部配置了,优先使用 filter_backends = [OrderingFilter] # 配置参与排序的字段 ?ordering=排序的关键字 # ?ordering=price,按价格升序 # ?ordering=-price,按价格降序 ordering_fiedls = ['price', 'id']
搜索组件 SearchFilter
搜索组件的源码过程和排序的一模一样都是在同一个py文件中,可以直接定位过来
- 在使用的视图类中导入排序类
SearchFilter
- 配置
filter_backends
- 配置参与排序的字段
search_fields
- (了解)可选配置:
search_param
规定接口中使用过滤的关键字
from . import models from rest_framework.generics import ListAPIView # 在使用的视图类中导入排序类 `OrderingFilter from rest_framework.filters import SearchFilter class FreeCourseListAPIView(ListAPIView): queryset = models.Course.objects.filter(is_delete=False, is_show=True, ).order_by('-orders').all() serializer_class = serializers.FreeCourseModelSerializer # 配置 搜索的过滤类,相当于是局部配置了,优先使用 filter_backends = [SearchFilter] # 配置参与搜索的字段 ?search=python search_fields = ['name']
自定义limit限制条件过滤器
我们基于以上的排序、搜索过滤器明白了它们是怎么实现的,如果我们想要自定义过滤类的话可以实现
- 继承了
BaseFilterBackend
类,自定义的过滤器可以不用继承父类,但是必须要实现filter_queryset
方法 - 重写
filter_queryset
方法 - 返回
queryset
对象
比如说我们现在就写一个限制返回多少条数据的过滤器
# 在当前的app下,新建一个.py文件 # 导入 BaseFilterBackend 基础类 from rest_framework.filters import BaseFilterBackend # 自定义的过滤器,继承 BaseFilterBackend 类 class LimitFilter(BaseFilterBackend): # 重写 `filter_queryset` 方法 def filter_queryset(self, request, queryset, view): # 从请求接口中拿到过滤的关键字 limit limit = request.query_params.get('limit') try: # queryset就是这句话: # queryset = models.User.objects.filter(is_delete=False, is_show=True).order_by('-orders').all() # 对结果切片 达到限制查询结果的功能 return queryset[:int(limit)] except: # 返回 `queryset` 对象 return queryset
# 视图类view中配置 from . import models from rest_framework.generics import ListAPIView # 导入我们自己定义的过滤类 from .myfilter import LimitFilter class FreeCourseListAPIView(ListAPIView): queryset = models.User.objects.filter(is_delete=False, is_show=True).order_by('-orders').all() serializer_class = serializer.RegisterModelSerializer # 将自己些定义的过滤类配置上就ok了 filter_backends = [LimitFilter]
筛选插件 djanog_filter
django和drf都实现不了分类,只能依赖第三方插件
pip install djanog_filter
源码分析
from django_filters.rest_framework import DjangoFilterBackend
进入DjangoFilterBackend,找到filter_queryset()方法
进入get_filterset()方法
进入get_filterset_class()方法
分类筛选 DjangoFilterBackend
用法:
# view.py视图类中 from . import models from rest_framework.generics import ListAPIView # 分类筛选 from django_filters.rest_framework import DjangoFilterBackend class FreeCourseListAPIView(ListAPIView): queryset = models.User.objects.filter(is_delete=False, is_show=True).order_by('-orders').all() serializer_class = serializer.RegisterModelSerializer # 分类筛选中要配置DjangoFilterBackend filter_backends = [DjangoFilterBackend] # 配置参与分类筛选的字段,所有字段都可以,一般用于分组的字段更有意义 # course_category 是外键 # ?course_category=1 得到外键course_category=1的数据 # 以下四种配置的关键字:都可以,因为在源码中做了兼容 filter_fields = ['course_category'] filterset_fields = ['course_category'] filter_class = ['course_category'] filterset_class = ['course_category']
区间筛选(自定义区间筛选类)
基于django-filter插件,完成指定区间筛选(一般都是对于数字字段)
- 新定义一个类,继承
FilterSet
- 定义配置类
Meta
- 视图类中配置使用
filter_class
# 创建一个filter.py文件 # 基于django-filter插件 from django_filters.rest_framework.filterset import FilterSet from . import models from django_filters import filters class CourseFilterSet(FilterSet): # 自定义限制字段,fileld_name:数据库中的字段,lookup_expr:设置条件 # 最大价格,lte小于等于 max_price = filters.NumberFilter(field_name='price', lookup_expr='lte') # 最小价格,gte大于等于 min_price = filters.NumberFilter(field_name='price', lookup_expr='gte') class Meta: model = models.Course fields = ['course_category', 'max_price', 'min_price']
from .filters import CourseFilterSet from . import models from rest_framework.generics import ListAPIView class FreeCourseListAPIView(ListAPIView): queryset = models.Course.objects.filter(is_delete=False, is_show=True, ).order_by('-orders').all() serializer_class = serializers.FreeCourseModelSerializer # 配置过滤类,优点是可以自定义区间过滤条件 # ?max_price=100&min_price=10 filter_class = CourseFilterSet
分页
进入到Mixins.ListModelMixin类中,我们会发现list方法内部在群查完得到结果以后,直接开始page配置了
由此可见分页过滤中有三种分页
- PageNumberPagination :普通分页
- LimitOffsetPagination : 显示条数分页,偏移
- CursorPagination :游标分页,和普通分页的区别在于,将url中前一页,后一页的直接加密
普通分页 PageNumberPagination
# 通过自定义分页的方法,来配置 # 在当前的apps下创建一个paginations.py的文件 from rest_framework.pagination import PageNumberPagination,LimitOffsetPagination,CursorPagination # 基础分页 class CoursePageNumberPagination(PageNumberPagination): # 配置条数,默认一页显示的条数 page_size = 2 # 查询页码的关键字,接口携带的参数 ?page=1 page_query_param = 'page' # 用户自定义一页显示的条数的关键字 page_size_query_param = 'page_size' # 用户可以自定义最大的一页显示的条数 max_page_size = 10
# view.py视图类中 from rest_framework.generics import ListAPIView from . import models, serializers # 自定义的普通分页 from .paginations import MyPageNumberPagination class FreeCourseListAPIView(ListAPIView): queryset = models.Course.objects.filter(is_delete=False, is_show=True).order_by('-orders').all() serializer_class = serializers.FreeCourseModelSerializer # 分页器 pagination_class = CoursePageNumberPagination
偏移分页 LimitOffsetPagination
from rest_framework.pagination import PageNumberPagination,LimitOffsetPagination,CursorPagination # 偏移分页:可以规定从第几条开始 class CourseLimitOffsetPagination(LimitOffsetPagination): # 默认一页条数 default_limit = 2 # 从offset开始往后显示limit条 limit_query_param = 'limit' offset_query_param = 'offset' # 用户可以自定义最大的一页显示的条数 max_limit = 2
# view.py视图类中 from rest_framework.generics import ListAPIView from . import models, serializers # 自定义的普通分页 from .paginations import MyLimitOffsetPagination class FreeCourseListAPIView(ListAPIView): queryset = models.Course.objects.filter(is_delete=False, is_show=True).order_by('-orders').all() serializer_class = serializers.FreeCourseModelSerializer # 分页器 pagination_class = CourseLimitOffsetPagination
游标分页 CursorPagination
from rest_framework.pagination import PageNumberPagination,LimitOffsetPagination,CursorPagination class CourseCursorPagination(CursorPagination): cursor_query_param = 'cursor' page_size = 2 page_size_query_param = 'page_size' max_page_size = 2 # ordering = 'id' # 默认排序规则,不能和排序过滤器OrderingFilter共存
# view.py视图类中 from rest_framework.generics import ListAPIView from . import models, serializers # 自定义的普通分页 from .paginations import MyCursorPagination class MyCursorPagination(ListAPIView): queryset = models.Course.objects.filter(is_delete=False, is_show=True).order_by('-orders').all() serializer_class = serializers.FreeCourseModelSerializer # 分页器 pagination_class = CourseCursorPagination