Django filter with OR condition using dict argument

前端 未结 3 1048
没有蜡笔的小新
没有蜡笔的小新 2020-12-09 05:50

I have a function on my Django app where I perform some Queryset actions and set it\'s result to Memcache. Since it is a function it has to be of general usage. So in order

相关标签:
3条回答
  • 2020-12-09 06:27

    You can use the bitwise | operator.

    my_filter = Q()
    
    # Or the Q object with the ones remaining in the list
    my_or_filters = {'some_field__gte':3.5, 'another_field':'Dick Perch'}
    
    for item in my_or_filters:
        my_filter |= Q(**{item:my_or_filters[item]})
    
    model.objects.filter(my_filter)
    # unpacks to model.objects.filter(Q(some_field__gte=3.5) | Q(another_field='Dick Perch'))
    

    With this in mind, you may want to load all your queries stored in my_filter into Q objects. Then, you could join all non-OR queries via the same method above w/ the bitwise &: my_filter &= ...

    0 讨论(0)
  • 2020-12-09 06:32

    You can restructure your dictionary into a list of single key-value dictionaries and use unpacking on each dict inside the Q expression like so:

    from functools import reduce
    import operator
    
    from django.db.models import Q
    
    # your dict is my_filter
    q = model.objects.filter(reduce(operator.or_, 
                                    (Q(**d) for d in [dict([i]) for i in my_filter.items()])))
    

    reduce on or_ joins the Q expressions on an OR.

    You could also use a generator expression where you have the list of dicts:

    q = model.objects.filter(reduce(operator.or_, 
                                    (Q(**d) for d in (dict([i]) for i in my_filter.items()))))
    
    0 讨论(0)
  • 2020-12-09 06:38

    Based on @Moses Koledoye's answer I could solve the issue. This is how my function looks like now:

     cached_query(key, model, my_filter=None, or_filter={}, exclude=None, order_by=None, sliced=50):
        """
        :param key: string used as key reference to store on Memcached
        :param model: model reference on which 'filter' will be called
        :param my_filter: dictionary containing the filter parameters (eg.: {'title': 'foo', 'category': 'bar'}
        :param or_filter: dictionary containing the filter parameters (eg.: {'title': 'foo', 'category': 'bar'}
        :param sliced: integer limit of results from the query. The lower the better, since for some reason Django Memcached
            won't store thousands of entries in memory
        :param exclude: dictionary containing the exclude parameters (eg.: {'title': 'foo', 'category': 'bar'}
        :param order_by: tuple containing the list of fields upon which the model will be ordered.
        :return: list of models. Not a QuerySet, since it was sliced.
        """
    
        result = cache.get(key, None)
        if not result:
            result = model.objects.all()
            if my_filter:
                result = model.objects.filter(**my_filter)
            if or_filter:
                reduced_filter = reduce(operator.or_, (Q(**d) for d in [dict([i]) for i in or_filter.items()]))
                result = result.filter(reduced_filter)
            if exclude:
                result = result.exclude(**exclude)
            if order_by:
                result = result.order_by(*order_by)
            result = result[:sliced]
            cache.set(key, result, cache_timeout)
        return result
    
    0 讨论(0)
提交回复
热议问题