在之前的一篇文章:【一日一技】Django不定项数的“或查询”里面,我们讲到,如果在Django的model中对多个查询条件使用 或
操作,那么代码可以这样写:
from django.db.models import Q keywords = ['促销', '打折', '限时'] item = Product.objects if keywords: filters = Q(description__contains=keywords[0]) for keyword in keywords[1:]: filters |= Q(description__contains=keyword) item = item.filter(filters)
不过这样写代码并不好看,因为要先把第一个参数取出来单独生成一个查询对象。然后再用这个查询对象与后面的参数形成的查询对象取或操作。
对于这种从一个可迭代对象里面依次读取每一个元素,传入到一个函数中,生成的结果再依次与可迭代对象后面的数据进行相同的操作,我们可以使用 reduce
。
例如有一个函数 func
,它接收两个参数,返回一个参数。现在我们有一个列表, [1,2,3,4,5]
,想实现:
a = [1, 2, 3, 4, 5] data = func(a[0], a[1]) data_2 = func(data, a[2]) data_3 = func(data_2, a[3]) result = func(data_3, a[4])
那么,代码可以这样修改:
from functools import reduce result = reduce(func, [1, 2, 3, 4, 5])
所以,对应到Django中不定项或查询,代码可以修改为:
from django.db.models import Q from functools import reduce keywords = ['促销', '打折', '限时'] def func(word_1, word_2): return Q(description__contains=word_1) | Q(description__contains=word_2) item = Product.objects if len(keywords) >= 2: filters = reduce(func, keywords) else: filters = Q(description__contains=keyowrds[0]) if keywords else None item = item.filter(filters)
这里需要注意,使用reduce的时候,需要保证它的第二个参数能至少被迭代2次。如果可迭代参数为空列表或者不能迭代的对象,那么就会导致报错。如果只能迭代1次,活着列表里面只有一个元素,那么就会直接返回这个元素,不会被传入函数中。只有当列表里面的元素不小于2个,或者可迭代对象能被迭代的次数不小于2次,reduce才能正常工作。