DRF搜索、分页、排序、过滤的实现

一、搜索

搜索

REST framework有提供了一个比较简洁的搜索功能给我们使用。

class GoodsListViewSet(viewsets.ModelViewSet):
    #数据
    queryset = Goods.objects.all()
    # 指定序列化的类
    serializer_class = GoodsSerializer
    #配置搜索功能
    filter_backends = (filters.SearchFilter,)
    #设置搜索的关键字
    search_fields=('name','goods_brief','goods_desc')

'''
        ^ :搜索关键字开头的数据
        = :完全匹配搜索                  =name :则是完全匹配,不能模糊搜索
        @ :全文搜索(目前只支持MySQL)
        $ :正则表达式搜索
'''

 

 在没有配置filter_backends之前,显示的效果如下:

DRF搜索、分页、排序、过滤的实现_第1张图片

 配置filter_backends之后:会出现一个过滤器

DRF搜索、分页、排序、过滤的实现_第2张图片

 

二、过滤

1、自定义过滤

class GoodsListViewSet(viewsets.ModelViewSet):
    serializer_class=GoodsSerializer
    #自定义过滤
    def get_query(self):
        price_min=self.request.query_params.get('price_min',0)
        if price_min:
            queryset=queryset.filter(shop_price__gt=int(price_min))
        return queryset

 2、使用django-filter搜索

首先安装django-filter:pip install django-filter  , 然后注册到app

自定义过滤类:

class GoodsFilter(django_filters.rest_framework.FilterSet):
    """
    商品的过滤类
    """
    pricemin = django_filters.NumberFilter(field_name='shop_price', help_text="最低价格", lookup_expr='gte')
    pricemax = django_filters.NumberFilter(field_name='shop_price', lookup_expr='lte')
    top_category = django_filters.NumberFilter(method='top_category_filter')

    def top_category_filter(self, queryset, name, value):
        return queryset.filter(Q(category_id=value) | Q(category__parent_category_id=value) | Q(
            category__parent_category__parent_category_id=value))

    class Meta:
        model = Goods
        fields = ['pricemin', 'pricemax', 'is_hot', 'is_new']

 views.py:

class GoodsListViewSet(viewsets.ModelViewSet):
    # 数据
    queryset = Goods.objects.all()
    # 指定序列化的类
    serializer_class = GoodsSerializer
    # 配置搜索功能
    filter_backends = (DjangoFilterBackend, filters.SearchFilter)
    # 设置搜索的关键字
    # filter_fields = ('name',)  # 默认是全匹配,可自定义
    filter_class = GoodsFilter  # 自定义过滤
    search_fields = ('name', 'goods_brief', 'goods_desc')  # 搜索
 

效果如下:

DRF搜索、分页、排序、过滤的实现_第3张图片

三、分页

1、全局分页

在settings.py文件中设置:

    REST_FRAMEWORK = {
        'DEFAULT_PAGINATION_CLASS': 'rest_framework.pagination.PageNumberPagination'
    }

 常见参数:

  1. PageNumberPagination类包括可以覆盖以修改分页样式的许多属性,要设置这些属性,应覆盖PageNumberPagination类,然后如上所示启用自定义分页类。
  2. django_paginator_class:使用的Django Paginator类,默认是django.core.paginator.Paginator,对大部分用例是适用的。
  3. page_size:数值,页面大小,默认是全局PAGE_SIZE的值。
  4. page_query_param:字符串,查询参数的名称,默认是'page'
  5. page_size_query_param:字符串,请求设置页面大小的参数名称,默认是None,表示客户端可能无法控制请求的页面大小。
  6. max_page_size:字符串,最大允许请求的页面大小, 此属性仅在page_size_query_param也被设置时有效。
  7. last_page_strings:字符串列表或者元组,默认是('last',)
  8. template:分页控件使用的模板的名称,可以覆盖或设置为None,默认为"rest_framework/pagination/numbers.html"

 2、自定义分页

# 定制分页
class GoodsPagination(PageNumberPagination):
    page_size = 12  # 每页显示多少个
    page_size_query_param = 'page_size'
    page_query_param = 'page'  # 查询参数的名称。默认是’page‘
    max_page_size = 100

views.py:

class GoodsListViewSet(viewsets.ModelViewSet):
    # 数据
    queryset = Goods.objects.all()
    # 指定序列化的类
    serializer_class = GoodsSerializer
    # 配置搜索功能
    filter_backends = (DjangoFilterBackend, filters.SearchFilter)
    # 设置搜索的关键字
    # filter_fields = ('name',)  # 默认是全匹配,可自定义
    filter_class = GoodsFilter  # 自定义过滤
    search_fields = ('name', 'goods_brief', 'goods_desc')
    pagination_class = GoodsPagination  # 自定义分页设置

DRF搜索、分页、排序、过滤的实现_第4张图片

 四、排序

在views.pywe的filter_backends添加 filters.OrderingFilter

class GoodsListViewSet(viewsets.ModelViewSet):
    # 数据
    queryset = Goods.objects.all()
    # 指定序列化的类
    serializer_class = GoodsSerializer
    # 配置搜索功能
    filter_backends = (DjangoFilterBackend, filters.SearchFilter, filters.OrderingFilter)
    # 设置搜索的关键字
    # filter_fields = ('name',)  # 默认是全匹配,可自定义
    filter_class = GoodsFilter  # 自定义过滤
    search_fields = ('name', 'goods_brief', 'goods_desc')
    pagination_class = GoodsPagination  # 自定义分页设置
    #添加排序字段
    ordering_fields = ('sold_num', 'shop_price')

 效果如下:
 

DRF搜索、分页、排序、过滤的实现_第5张图片

你可能感兴趣的:(Python,Road,DRF)