Django QuerySet联合运算符在注释过滤器之后是不可交换的



。。。并返回意外结果(在Django 1.6.5中)

我的型号.py

class Member(models.Model):
    ...
class Donation(models.Model):
    year = models.PositiveSmallIntegerField()
    cheque_amount = models.DecimalField(default=0, max_digits=8, decimal_places=2)
    donor = models.ForeignKey(Member)
    ...
class SpecialTitle(models.Model):
    chair_title = models.CharField(max_length=128, blank=True)
    member = models.OneToOneField(Member)
    ...

我想在我的一个管理过滤器中合并两个查询集

donors = queryset.filter(
    donation__year__exact=2014
).annotate(sum_donation=Sum('donation__cheque_amount')).filter(sum_donation__gte=1000)
chairs = queryset.filter(specialtitle__chair_title__iendswith='Chair')

以下是令人费解的部分(在Django管理器外壳中)

>>> donors | chairs == chairs | donors
False
>>> donors.count(); chairs.count()
189
17
>>> (donors | chairs).count(); (chairs | donors).count()
193
291
>>> (donors | chairs).distinct().count(); (chairs | donors).distinct().count()
193
207

没有一个是正确的结果。我希望一个固定的操作是

>>> set(donors) | set(chairs) == set(chairs) | set(donors)
True
>>> set(donors) & set(chairs) == set(chairs) & set(donors)
True
>>>

他们会返回正确的结果。然而,Django管理过滤器需要QuerySet,而不是python集(或列表)

为什么会这样?如何在注释过滤器之后获得Django QuerySet(相同类型)的正确并集?非常感谢。

我似乎别无选择,只能使用python集联合运算符并再次访问数据库以获得所需结果。

donors = queryset.filter(
    donation__year__exact=2014
).annotate(sum_donation=Sum('donation__cheque_amount')).filter(sum_donation__gte=1000)
chairs = queryset.filter(specialtitle__chair_title__iendswith='Chair')
result = queryset.filter(pk__in=[person.id for person in set(donors) | set(chairs)])

最新更新