【问题标题】:Simplify my Django Query Logic简化我的 Django 查询逻辑
【发布时间】:2011-10-07 18:29:28
【问题描述】:

我有一张像这样的桌子。

+----+--------+---------+-----------+------------------------+
| id | parent | type    | libTypeId | name                   |
+----+--------+---------+-----------+------------------------+
|  2 |      1 | project |         2 | 1p6m4x0y1z_1.2-1.8_hvt |
|  5 |      1 | project |         6 | 1p6m4x0y1z_1.2-1.8_hvt |
|  8 |      1 | project |        16 | 1p6m4x0y1z_1.2-1.8_hvt |
| 11 |      1 | project |        21 | 1p6m4x0y1z_1.2-1.8_hvt |
|  3 |      1 | project |         2 | 1p6m4x0y1z_1.2-1.8_lvt |
|  6 |      1 | project |         6 | 1p6m4x0y1z_1.2-1.8_lvt |
|  9 |      1 | project |        16 | 1p6m4x0y1z_1.2-1.8_lvt |
| 12 |      1 | project |        21 | 1p6m4x0y1z_1.2-1.8_lvt |
|  1 |      1 | project |         2 | 1p6m4x0y1z_1.2-1.8_svt |
|  4 |      1 | project |         6 | 1p6m4x0y1z_1.2-1.8_svt |
|  7 |      1 | project |        16 | 1p6m4x0y1z_1.2-1.8_svt |
| 10 |      1 | project |        21 | 1p6m4x0y1z_1.2-1.8_svt |
| 13 |      2 | project |         2 | 065nm_GPIO             |
| 17 |      2 | project |         4 | 065nm_GPIO             |
| 14 |      2 | project |         6 | 065nm_GPIO             |
| 18 |      2 | project |        12 | 065nm_GPIO             |
| 15 |      2 | project |        16 | 065nm_GPIO             |
| 16 |      2 | project |        21 | 065nm_GPIO             |
| 19 |      2 | project |         2 | 065nm_Specialized      |
+----+--------+---------+-----------+------------------------+

我正在寻找的是一个查询,它会生成一个列表,我们可以在其中获取所有 id = 1 的项目,该列表仅按名称、libtypeid 和 FIRST libtypeid 排序。

换句话说,我应该这样结束:

+----+--------+---------+-----------+------------------------+
| id | parent | type    | libTypeId | name                   |
+----+--------+---------+-----------+------------------------+
|  2 |      1 | project |         2 | 1p6m4x0y1z_1.2-1.8_hvt |
|  3 |      1 | project |         2 | 1p6m4x0y1z_1.2-1.8_lvt |
|  1 |      1 | project |         2 | 1p6m4x0y1z_1.2-1.8_svt |
+----+--------+---------+-----------+------------------------+

现在我可以做到这一点,但我怎么才能只得到第一个呢?

Variant.objects.filter(parent=self.id).order_by('name', 'libtype_id')

然后我进一步这样做..

full = Variant.objects.filter(parent=self.id).order_by('name', 'libtype_id')
names, out = [], []
for v in full:
    if v.name not in names:
       out.append(v)
       names.append(v.name)
return out

如果有人可以清理一下,非常感谢..

【问题讨论】:

  • 不清楚您要完成什么。您想从结果集中获取第一项吗?只需使用索引。
  • akonsu - 是的,我想要这组中的第一个唯一“名称”——我最终应该得到三个项目。你能给我一个例子吗?谢谢
  • (Variant.objects.filter(parent=self.id).order_by('name', 'libtype_id')[0]).name 是你要问的吗?我还是不明白……
  • 你是这个意思吗? Variant.objects.filter(parent=self.id).order_by('name', 'libtype_id')[0]
  • 不,我不想要完整集的第一条记录。我想要每个子集的第一条记录。第一个集合的分组方式是按名称,后跟 libtypeid。我只想要第一个 libtypeid 而不是全部。

标签: mysql django django-queryset


【解决方案1】:

如果我正确理解了这个问题,您需要“分组”您的集合并返回每个组的第一行。你需要看看聚合:https://docs.djangoproject.com/en/dev/topics/db/aggregation/

类似的东西

Variant.objects
    .values('name', 'libtype_id')
    .annotate(min_libtype_id=Min('libtype_id'))
    .filter(parent=self.id)
这是未经测试的

【讨论】:

    【解决方案2】:

    您可以将[number] 添加到查询集中,例如:

    Variant.objects.filter(parent=self.id).order_by('name', 'libtype_id')[0]
    

    返回查询集中的第一条记录。要得到最后一个,应该是[:0],冒号反转拼接。

    【讨论】:

    • 不,这是集合的第一个元素。我想要集合的所有第一个子元素。
    【解决方案3】:

    这是我一直在寻找的答案..

    class VariantQuerySet(QuerySet):
        def as_single_library_type(self):
            query = self.filter().query
            query.group_by = ['name']
            return QuerySet(self.model, using=self._db, query=query)
    
    class VariantManager(models.Manager):
    
        def get_query_set(self):
            return VariantQuerySet(self.model, using=self._db)
    

    这基本上是默认的GROUP BY,它将所有内容都减少到我需要的范围内。感谢所有尝试回答的人!

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2014-03-25
      • 1970-01-01
      • 1970-01-01
      • 2011-08-17
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多