三十五、DRF中的过滤、Django中的过滤后端

一、DRF中的过滤

REST framework 的通用列表视图的默认行为是返回模型管理器的整个查询集。通常,您希望 API 限制查询集返回的项目。

过滤子类化的任何视图的查询集的最简单方法是覆盖GenericAPIView.get_queryset() 方法。覆盖此方法允许您以多种不同方式自定义视图返回的查询集。

(1)针对当前用户过滤

您可能希望过滤查询集以确保只返回与当前已通过身份验证的发出请求的用户相关的结果。您可以通过request.user 的值进行过滤来实现。

from rest_framework import viewsets
from rest_framework.permissions import IsAuthenticated 
from rest_framework.pagination import PageNumberPagination
from projects import serializers from projects import models
from projects.permissions import IsOwnerOrReadOnly, IsLeaderCreateOnly


class ProjectViewSet(viewsets.ModelViewSet): 
    """project视图集"""
	serializer_class = serializers.ProjectSerializer 
	permission_classes = [IsAuthenticated, IsOwnerOrReadOnly]

    def get_queryset(self)
    	return models.Project.objects.filter(leader=self.request.user)

(2)根据URL 过滤(路径参数)

另一种过滤方式可能涉及基于URL 的某些部分限制查询集。例如,如果您的URL 配置包含这样的条目:

re_path('^projects/(?P<username>.+)/$', ProjectViewSet.as_view()),

然后,您可以编写一个视图,该视图返回由URL 的用户名部分过滤的项目查询集:

def get_queryset(self):
    username = self.kwargs['username']
    return models.Project.objects.filter(leader username=username)

 

(3)根据查询参数过滤

过滤初始查询集的最后一个示例是根据url 中的查询参数确定初始查询集。

我们可以覆盖.get_queryset() 以处理诸如 http://example.com/api/purchases?username=denvercoder9之类的URL,并且仅当URL 中包含username 参数时才过滤查询集:

def get_queryset(self):
    queryset = models.Project.objects.all()
    username = self.request.query_params.get('username')
    if username is not None:
		queryset = queryset.filter(username=username)
	return queryset

(4)通用过滤

除了能够覆盖默认查询集之外,REST 框架还包括对通用过滤后端的支持,允许您轻松构建复杂的搜索和过滤器。

4.1 设置过滤器后端

可以使用设置全局设置默认过滤器后端DEFAULT_FILTER_BACKENDS 。例如。

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

 您还可以使用GenericAPIView 基于类的视图在每个视图或每个视图集的基础上设置过滤器后端。

import django_filters.rest_framework
from django.contrib.auth.models import User 
from myapp.serializers import UserSerializer 
from rest_framework import generics

class UserListView(generics.ListAPIView): 
    queryset = User.objects.all() 
    serializer_class = UserSerializer
    filter_backends = [django_filters.rest_framework.DjangoFilterBackend]
	# filterset_fields = ['project', 'type']

 一般情况下我们不需要自己实现过滤后端,有很多很成熟的三方插件供我们选择。我们这里使用django-filter 。

二、Django过滤后端

(1) 安装

django-filter库包括一个支持 REST 框架的高度可定制的字段过滤的类 DjangoFilterBackend 。 要使用 DjangoFilterBackend ,请先安装 django-filter 。

pip install django-filter

注意 djoango-filter 仅支持:

Python: 3.6, 3.7, 3.8

Django: 2.2, 3.1, 3.2

DRF: 3.10+

(2)配置

然后添加 'django_filters' 到 Django 的 INSTALLED_APPS:

INSTALLED_APPS = [
...
'django_filters',
...
]

您现在应该将过滤器后端添加到您的设置中:

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

或者将过滤器后端添加到单个视图或视图集。

from django_filters.rest_framework import DjangoFilterBackend
class UserListView(generics.ListAPIView):
...
filter_backends = [DjangoFilterBackend]

(3)使用

在后端类视图或者视图集中添加 filterset_fields属性即可实现指定字段过滤,例如改写环境管理视图集如下:

class TestEnvViewSet(ModelViewSet):
"""测试环境视图集"""
    queryset = TestEnv.objects.all()
    serializer_class = TestEnvSerializer
    filterset_fields = ('project', )
    permission_classes = [IsAuthenticated]

不需要在复写 get_queryset方法手动过滤,就可以通过url/test_envs/?project=1 访问过滤数据了。

注意:当使用外键字段过滤时,如果级联模式是删除,则使用不存在的数据过滤会返回400响应。例如上面的视图如 果使用不存在的项目id过滤返回结果如下:

# http://127.0.0.1:8000/environments/?project=2
{
"project": [
"选择一个有效的选项:    该选择不在可用的选项中。  "
]
}

filters.py文件

在应用中新建一个过滤器得模块,来写过滤器:支持条件(模糊等)查询


from django_filters import rest_framework as filters
from .models import Interface


class InterFaceFilter(filters.FilterSet):
    """"""
    # 字段名是查询参数名
    project_name_contains = filters.CharFilter(field_name='project__name', lookup_expr='contains')
    # Interface.objects.filter(project__name__contains='百度')

    class Meta:
        model = Interface
        # 过滤字段
        fields = ['project', 'type']

views里使用

class InterfaceViewSet(ModelViewSet):
    serializer_class = InterfaceSerializer
    queryset = Interface.objects.all()
    permission_classes = [IsAuthenticated, IsLeaderOrReadOnly]
    # filterset_fields = ['project', 'type']
    filterset_class = InterFaceFilter

更多详细用法见官方文档: https://django-filter.readthedocs.io/en/latest/guide/rest_framework.html

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值