django - 改进迭代查询

Posted

技术标签:

【中文标题】django - 改进迭代查询【英文标题】:django - improving iterative queries 【发布时间】:2016-02-04 15:47:58 【问题描述】:

我有一个 django 应用程序,我在其中迭代地循环执行过滤器。 这是一个简化的例子:

class Decision(models.Model):
    recommendation = models.TextField()

class Condition(models.Model):
    dec = models.ForeignKey(Decision, related_name='condition')
    temperature = models.PositiveInteger()
    pressure = models.PositiveInteger()

Decision.objects.filter(condition__temperature=22, condition__pressure=123 ).filter(condition__temperature=30, condition__pressure=144).values_list('id',flat=True)

如您所见,条件是“与”。 有没有更有效的方法来做这个查询?

这是另一种方法,但它没有给我任何结果:

Decision.objects.filter(Q(condition__temperature=22, condition__pressure=123 ) &\
    Q(condition__temperature=30, condition__pressure=144)).values_list('id',flat=True)

【问题讨论】:

我有点惊讶第一个查询显然给了你结果;正如您所说,我认为过滤器链是AND-ed,因此第一个过滤器排除了第二个过滤器(因为condition__temperature=22 排除了condition_temperature=30)。这就是你确实(正确地,我认为)在你的其他方法中得到的。问题可能是:结合两个独占条件,您要达到什么目的? 另见***.com/questions/8164675/…。过滤器链确实是 OR-ed 在一起的,因此要在第二种方法中获得相同的行为,请将 & 替换为 |。如果这确实是你想要的。 【参考方案1】:

您的查询几乎是正确的,但它必须是OR

Decision.objects.filter(
    Q(condition__temperature=22, condition__pressure=123 ) |  # OR
    Q(condition__temperature=30, condition__pressure=144)
).values_list('id',flat=True)

很明显,你的条件的温度不能同时是22AND30。

你可以对条件列表做同样的事情:

from operator import or_
from django.db.models import Q

predicates = [
    'condition__temperature': 22, 'condition__pressure': 123, 
    'condition__temperature': 30, 'condition__pressure': 144]
conditions = [Q(x) for x in predicates]
query = Decision.objects.filter(reduce(or_, conditions))

您可以在此处阅读有关 Q 的完整文档:django-docs

【讨论】:

从 Condition 到 Decision 有一个外键,这意味着您可以有 2 个温度(在不同的位置)。也许我的例子不好,但问题的棘手部分是这是一个一对多的关系,所以你的建议对我不起作用。不过还是谢谢。

以上是关于django - 改进迭代查询的主要内容,如果未能解决你的问题,请参考以下文章

如何过滤 Django 查询的连接表,然后在一个查询中迭代连接表?

面向对象大作文第一次迭代改进任务

迭代 Django 中的相关对象:循环查询集或使用单行 select_related(或 prefetch_related)

Django - 在模板的for循环中迭代数字

改进的集合迭代器

手把手,带你用数据做好迭代复盘改进 | 敏捷开发落地指南