简化我的Django查询逻辑

时间:2011-10-07 18:29:28

标签: mysql django django-queryset

我有一张看起来像这样的表。

+----+--------+---------+-----------+------------------------+
| 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

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

3 个答案:

答案 0 :(得分: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)
这是未经测试的

答案 1 :(得分:0)

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

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

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

答案 2 :(得分:0)

以下是我正在寻找的答案..

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,可以将所有内容减少到我需要的范围。感谢所有尝试回答的人!