如果在Django的model中对多个查询条件使用 操作,那么代码可以这样写:

  1. from django.db.models import Q

  2.  

  3. keywords = ['促销', '打折', '限时']

  4.  

  5. item = Product.objects

  6.  

  7. if keywords:

  8.  

  9. filters = Q(description__contains=keywords[0])

  10.  

  11. for keyword in keywords[1:]:

  12.  

  13. filters |= Q(description__contains=keyword)

  14.  

  15. item = item.filter(filters)

不过这样写代码并不好看,因为要先把第一个参数取出来单独生成一个查询对象。然后再用这个查询对象与后面的参数形成的查询对象取或操作。

对于这种从一个可迭代对象里面依次读取每一个元素,传入到一个函数中,生成的结果再依次与可迭代对象后面的数据进行相同的操作,我们可以使用 reduce

例如有一个函数 func,它接收两个参数,返回一个参数。现在我们有一个列表, [1,2,3,4,5],想实现:

  1. a = [1, 2, 3, 4, 5]

  2. data = func(a[0], a[1])

  3. data_2 = func(data, a[2])

  4. data_3 = func(data_2, a[3])

  5. result = func(data_3, a[4])

那么,代码可以这样修改:

  1. from functools import reduce

  2. result = reduce(func, [1, 2, 3, 4, 5])

所以,对应到Django中不定项或查询,代码可以修改为:

  1. from django.db.models import Q

  2. from functools import reduce

  3.  

  4. keywords = ['促销', '打折', '限时']

  5.  

  6. def func(word_1, word_2):

  7. return Q(description__contains=word_1) | Q(description__contains=word_2)

  8.  

  9. item = Product.objects

  10.  

  11. if len(keywords) >= 2:

  12. filters = reduce(func, keywords)

  13. else:

  14. filters = Q(description__contains=keyowrds[0]) if keywords else None

  15. item = item.filter(filters)

这里需要注意,使用reduce的时候,需要保证它的第二个参数能至少被迭代2次。如果可迭代参数为空列表或者不能迭代的对象,那么就会导致报错。如果只能迭代1次,活着列表里面只有一个元素,那么就会直接返回这个元素,不会被传入函数中。只有当列表里面的元素不小于2个,或者可迭代对象能被迭代的次数不小于2次,reduce才能正常工作。