Python Django-如何根据外键模型关系排除queryset中的实例

Python Django-如何根据外键模型关系排除queryset中的实例,python,django,Python,Django,如果有任何项目实例不在某个项目列表中,我将尝试筛选报价模型中的实例。从ProjectItem模型中检索特定的项列表。下面是模型结构,我将在代码之后解释更多内容 models.py class SalesProject(models.Model): sales_project_id = models.AutoField(primary_key=True) sales_project_name = models.CharField(max_length=100) class Pro

如果有任何项目实例不在某个项目列表中,我将尝试筛选报价模型中的实例。从ProjectItem模型中检索特定的项列表。下面是模型结构,我将在代码之后解释更多内容

models.py

class SalesProject(models.Model):
    sales_project_id = models.AutoField(primary_key=True)
    sales_project_name = models.CharField(max_length=100)

class ProjectItem(models.Model):
    project_item_id = models.AutoField(primary_key=True)
    project = models.ForeignKey('SalesProject', related_name='items', on_delete=models.CASCADE)
    item = models.ForeignKey('Item', on_delete=models.CASCADE)
    remarks = models.TextField(max_length=1000)

class Quotation(models.Model):
    quotation_id = models.AutoField(primary_key=True)
    salesProject = models.ForeignKey(
        'SalesProject', related_name='quotations', on_delete=models.CASCADE, null=True, blank=True)
    details = models.TextField(max_length=1000, blank=True, null=True)

class QuotationItem(models.Model):
    quotation_item_id = models.AutoField(primary_key=True)
    item = models.ForeignKey('Item', on_delete=models.CASCADE, null=True, blank=True)
    quotation = models.ForeignKey('Quotation', on_delete=models.CASCADE, related_name='items', null=True, blank=True)
    remarks = models.TextField(max_length=1000)

class Item(models.Model):
    item_id = models.AutoField(primary_key=True)
    item_code = models.CharField(max_length=500, null=True, blank=True)
    item_description = models.TextField(max_length=1000, null=True, blank=True)
首先,我将通过查询当前SalesProject来获得Item实例的列表。(其中project=SalesProject实例)

基本上,此项目列表中的项目实例是各种“允许的项目”。我想返回的是此列表中仅包含项目实例的所有报价实例(或排除此列表之外包含项目实例的任何报价实例)。报价与项目的关系是通过QuotationItem模型实现的。有什么办法吗

感谢所有人的帮助,一定要指导我,如果信息不足,请告诉我。

您可以使用以下方法进行筛选:

Quotation.objects.filter(items__item__projectitem__project=project)
如果您想要只引用
项目的
报价
,可以使用
.exclude(…)
,或者更直接地计算
项目

from django.db.models import Count, Q

Quotation.objects.annotate(
    nproject=Count('items__item__projectitem__project', distinct=True),
    nproject_project=Count(
        'items__item__projectitem__project',
        distinct=True,
        filter=Q(items__item__projectitem__project=project)
    ),
).filter(
    nproject_project=1,
    nproject=1
).distinct()
从django.db.models导入计数,Q
quote.objects.annotate(
nproject=Count('items\uuu item\uuuu projectitem\uuuu project',distinct=True),
nproject_project=计数(
“项目项目项目”,
独特=正确,
过滤器=Q(项目项目项目项目项目=项目)
),
).过滤器(
nproject_project=1,
n项目=1
).distinct()

注意:您的
QuotationItem
模型基本上就像, 您可能希望添加一个
ManyToManyField
,将此模型作为一个示例。 这使得使用Django ORM查询此类关系更加方便


引号
如何准确地映射到
项目项
?乍一看,这两者仅通过
Porject
连接。谢谢你,它成功了!但是我可以检查一下,您是如何在筛选器查询中获取projectitem的?是否使用默认的反向名称?(从项目到项目的关系)@jason:通常使用
quote.objects.exclude(~Q(items\uu Item\uuu projecttem\uu project=project)).filter(items\uu Item\uuuuu projecttem\uu project=project)
(其中您可以从django.db导入
Q
)。@jason:这是对条件的否定,因此,我们希望排除所有不引用project的对象。@jason:当然不是这样,因为可能根本就没有
QuotationItem
,在这种情况下,排除不会排除这些对象。@jason:您也可以使用聚合,请参阅编辑。
Quotation.objects.filter(items__item__projectitem__project=project).distinct()
from django.db.models import Count, Q

Quotation.objects.annotate(
    nproject=Count('items__item__projectitem__project', distinct=True),
    nproject_project=Count(
        'items__item__projectitem__project',
        distinct=True,
        filter=Q(items__item__projectitem__project=project)
    ),
).filter(
    nproject_project=1,
    nproject=1
).distinct()