Django-Filter是一个非常好用的第三方库,很好的利用了Django ORM的特性,可以使用很少的代码就扩展原有的接口,实现多种筛选功能~
场景
Model定义的部分代码,可以看到需求模型包括了 关键词 keyword
、区域 region
、需求状态 require_status
,这三个字段
class Require(models.Model):
"""需求"""
keyword = models.CharField('需求关键字', max_length=1000, blank=True)
region = models.ForeignKey(Region, verbose_name='落地实施区域', null=True, on_delete=models.SET_NULL)
require_status = models.IntegerField('需求进行状态', choices=RequireStatusChoice.choices)
之后是ViewSet的代码,在Drf中经过简单的路由配置就可以获取需求列表了,但是真实场景要求的功能肯定不止这些
class RequireViewSet(viewsets.ModelViewSet):
"""需求记录"""
serializer_class = RequireSerializer
queryset = Require.objects.all()
产品需要的功能很多,针对这个需求模块而言,需要:
- 根据关键词搜索需求
- 根据落地实施区域筛选需求
- 根据需求进行状态筛选需求
如果现在我们用的是Java语言,那肯定只能老老实实去写多几行代码来实现,但是!「人生苦短,我用Python!」
用Python的我们岂能屈服于一个简单的功能实现,该偷懒的时候必须偷懒,没办法偷懒也得强行偷懒hhh~
Django-Filter的简单使用方法
看看使用Django-Filter我们如何实现,首先是安装Django-Filter:
pip install django-filter
添加到settings.py
中:
INSTALLED_APPS = [
...
'django_filters',
]
然后在ViewSet
里面配置就好了:
from django_filters.rest_framework import DjangoFilterBackend
class RequireViewSet(viewsets.ModelViewSet):
"""需求记录"""
serializer_class = RequireSerializer
queryset = Require.objects.all()
filter_backends = [DjangoFilterBackend, SearchFilter]
# 参与分类筛选的字段
filter_fields = ['require_status', 'region']
# 参与搜索的字段: search=关键词
search_fields = ['keyword']
配置之后,就可以在原有基础上进行筛选和搜索了,假设接口地址是:http://127.0.0.1/require/
搜索
http://127.0.0.1/require/?search=关键词
按照区域筛选
http://127.0.0.1/require/?region=1
按照需求状态筛选
http://127.0.0.1/require/?require_status=2
效果很完美,只是简单的配置就可以实现这么多功能,这就是偷懒的好处(站在巨人的肩膀上),其实这些CRUD的功能没啥技术含量,框架和第三方库的作者都给我们实现好了这些功能,直接拿来用就行了~
不过这时前端小伙伴又提出了新的要求,就是 需求状态 require_status
需要同时筛选多个状态的需求,比如同时筛选状态编号为2-9的需求…… 这可就难搞了,Django-Filter默认只支持一个啊,怎么办啊……
但是别急,我相信任何问题都难不倒我Stack Overflow,所以我几个关键词一通搜索,试了热心网友提供的四五种方法,果然找到了最佳解决方法…
多个值同时筛选
首先要定义一个自定义的Filter,这里我使用逗号来分割每个参数,代码如下:
from django_filters import Filter,FilterSet
class ListFilter(Filter):
def filter(self, qs, value):
if not value:
return qs
# For django-filter versions < 0.13, use lookup_type instead of lookup_expr
self.lookup_expr = 'in'
values = value[0:1000].split(',')
return super(ListFilter, self).filter(qs, values)
class RequireStatusFilter(FilterSet):
require_statuses = ListFilter(field_name='require_status')
class Meta:
from apps.require.models import Require
model = Require
fields = ['require_statuses', 'region']
然后修改我们的ViewSet,可以看到不需要filter_fields
了,因为在RequireStatusFilter
里面已经定义好了
class RequireViewSet(viewsets.ModelViewSet):
"""需求记录"""
serializer_class = RequireSerializer
queryset = Require.objects.all()
permission_classes = [permissions.IsAuthenticated]
filter_backends = [DjangoFilterBackend, SearchFilter]
filter_class = RequireStatusFilter
search_fields = ['keyword']
使用方法
美滋滋简直
http://127.0.0.1/require/?region=1&require_status=1,2,3,4,5,6,7,8,9
吃瓜
最后吃个瓜,今天Python之父发推说加入微软了,Delphi/C#/TypeScript之父在评论区发了贺电,哈哈哈
参考
- Stack Overflow问题:
- 官方文档: