你如何使用带有参数列表的 django-filter 包?

Posted

技术标签:

【中文标题】你如何使用带有参数列表的 django-filter 包?【英文标题】:How do you use the django-filter package with a list of parameters? 【发布时间】:2015-09-10 20:19:07 【问题描述】:

我想用django-filter 过滤我的模型。如果我按一个 id 过滤,它工作正常:

http://localhost:8000/accommodations?accommodationType_id=1

但我不知道如何按多个 id 进行过滤。

http://localhost:8000/accommodations?accommodationType_id=1,2

我的实际ViewSet 是这样的:

class AccommodationViewSet(viewsets.ReadOnlyModelViewSet):
    """
        REST API endpoint for 'accommodation' resource
    """
    queryset = Accommodation.objects.all()
    serializer_class = AccommodationSerializer
    filter_backends = (filters.DjangoFilterBackend,)
    filter_fields = ('accommodationType_id', 'name')

希望有解决办法。

【问题讨论】:

你试过了吗:http://localhost:8000/accommodations?accommodationType_id=1&accommodationType_id=2 是的,我得到了最后一个带有 accommodationType_id=2 的对象 【参考方案1】:

我为我的问题找到了以下解决方案:)

https://gist.github.com/aBuder/654fb945f085b17358d8

from webapp.serializers import *
from rest_framework import viewsets
from rest_framework import filters
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.split(',')
        return super(ListFilter, self).filter(qs, values)


class AccommodationFilter(FilterSet):
    ids = ListFilter(name='id')
    accommodationType_ids = ListFilter(name='accommodationType_id')

    class Meta:
        model = Accommodation
        fields = ['ids', 'accommodationType_ids']


class AccommodationViewSet(viewsets.ReadOnlyModelViewSet):
    """
        REST API endpoint for 'accommodation' resource
    """
    queryset = Accommodation.objects.all()
    serializer_class = AccommodationSerializer
    filter_backends = (filters.DjangoFilterBackend,)
    filter_class = AccommodationFilter

【讨论】:

谢谢!这对我帮助很大。如果您现在有任何其他提示,我将非常感激!我需要经常根据列表获取项目【参考方案2】:

我知道这是一个老问题,但提供更新的答案可能是值得的。

Django-filter 贡献者添加了一个名为 BaseInFilter 的字段,您可以将其与其他过滤器结合使用以验证内容。

查看文档: https://django-filter.readthedocs.io/en/latest/ref/filters.html#baseinfilter

例如,这适用于您的情况:

from django_filters import rest_framework as filters


class NumberInFilter(filters.BaseInFilter, filters.NumberFilter):
    pass


class AccommodationFilter(filters.FilterSet):
    accommodationType_id_in = NumberInFilter(field_name='accommodationType_id', lookup_expr='in')

    class Meta:
        model = Accommodation
        fields = ['accommodationType_id_in', ]

然后您就可以通过 id 列表进行过滤:http://localhost:8000/accommodations?accommodationType_id_in=1,2

【讨论】:

非常有用,很棒 或者更简单,django-filter 将“字段”作为字典,您可以在其中指定查找。 django-filter.readthedocs.io/en/stable/ref/…首先在这个答案中找到它:***.com/questions/65020090/…【参考方案3】:

现在有一种更简单的方法可以实现这一点。在django-filter documentation 的深处,它提到您可以使用“映射到查找列表的字段名称字典”。

您的代码将像这样更新:

class AccommodationViewSet(viewsets.ReadOnlyModelViewSet):
    """
        REST API endpoint for 'accommodation' resource
    """
    queryset = Accommodation.objects.all()
    serializer_class = AccommodationSerializer
    filter_backends = (filters.DjangoFilterBackend,)
    filter_fields = 
        'accommodationType_id': ["in", "exact"], # note the 'in' field
        'name': ["exact"]
    

现在在 URL 中,您可以在提供参数列表之前将 __in 添加到过滤器中,它会按您的预期工作:

http://localhost:8000/accommodations?accommodationType_id__in=1,2

关于可用查找过滤器的django-filter 文档很差,但Django documentation 本身提到了in 查找过滤器。

【讨论】:

以上是关于你如何使用带有参数列表的 django-filter 包?的主要内容,如果未能解决你的问题,请参考以下文章

Django-Filter 包:如何过滤对象以创建统计信息而不是列表

Django-filters:单个查询字符串中的多个 ID

如何在 ListAPIView 中使用 django-filter 对过滤结果进行排序

如何在ListAPIView中使用django-filter对过滤结果进行排序

django-filter 使用分页

django-filter使用分页