Python 如何过滤Django查询集';s相关字段'';全部';或';无';

Python 如何过滤Django查询集';s相关字段'';全部';或';无';,python,django,django-queryset,django-orm,Python,Django,Django Queryset,Django Orm,例如,我有一个型号Cat,它有一把Life的反向ForeignKey class Life(models.Model): state = models.CharField(choices=('alive', 'dead', 'unknown') cat = models.ForeignKey('animals.Cat', related_name="lives") class Cat(models.Model): name = models.CharField(ma

例如,我有一个型号
Cat
,它有一把
Life
的反向
ForeignKey

class Life(models.Model):
    state = models.CharField(choices=('alive', 'dead', 'unknown')
    cat = models.ForeignKey('animals.Cat', related_name="lives")


class Cat(models.Model):
     name = models.CharField(max_length=12)
     cat_type = models.CharField(choices=('normal', 'schroedinger')
     ...
我如何获得一个猫的
查询集
,这些猫没有失去任何生命?也就是说,他们的所有生命要么处于“活着”状态,要么属于cat_类型的“薛定谔”,而他们的任何生命都没有处于“死亡”状态)

新答案:

cats = Cat.objects.filter(id=-1) # make it an empty queryset

temp = Cat.objects.filter(cat_type='normal')
for one in temp:
    if len(one.lives.exclude(state='alive')) == 0:
        cats |= Cat.objects.filter(id=one.id)

temp = Cat.objects.filter(cat_type='schroedinger')
for one in temp:
    if len(one.lives.filter(state='dead')) == 0:
        cats |= Cat.objects.filter(id=one.id)

return cats

我已经有一段时间没有使用此API了,但我相信这将完成工作:

from django.db.models import Q

normal_and_alive = Q(cat_type="normal") & ~Q(lives__state__in=["dead", "unknown"])
schroedinger_and_not_dead = Q(cat_type="schroedinger") & ~Q(lives__state="dead")

cats = Cat.objects.filter(normal_and_alive | schroedinger_and_not_dead)
有关django的文档,请参见文档


旁白:这将只执行一个数据库查询

我不确定,可能是cats+3的数量如果将state字段移动到Cat,命中率可能会低得多此代码将不会运行,
one.lifes.filter(…)
返回一个
Life()
记录的查询集,因为它本质上是
Cat.objects.all()| Life.objects.all()
。它不起作用,因为它们处于不同的位置tables@Jiaaro哦,是的,你是对的,我之所以被选中,仅仅是因为我懂一点物理知识来理解提问者想要什么。
from django.db.models import Q

normal_and_alive = Q(cat_type="normal") & ~Q(lives__state__in=["dead", "unknown"])
schroedinger_and_not_dead = Q(cat_type="schroedinger") & ~Q(lives__state="dead")

cats = Cat.objects.filter(normal_and_alive | schroedinger_and_not_dead)