sqlalchemy - 对项目进行分组并迭代子列表

时间:2009-08-14 18:18:31

标签: python sql sqlalchemy

考虑这样一个表:

| Name       | Version | Other |
| ---------------------|-------|
| Foo        | 1       | 'a'   |
| Foo        | 2       | 'b'   |
| Bar        | 5       | 'c'   |
| Baz        | 3       | 'd'   |
| Baz        | 4       | 'e'   |
| Baz        | 5       | 'f'   |
--------------------------------

我想编写一个sqlalchemy查询语句来列出所有项目(作为mapper对象,而不仅仅是Name列),最大版本为Foo-2-b, Bar-5-c, Baz-5-f。我知道我必须使用group_by方法,但除此之外,我对如何检索子列表(然后找到max元素)感到困惑。 SQLAlchemy文档显然不是很清楚。

在真实场景中,还有很多其他列(比如'Other') - 这就是为什么我需要返回实际的行对象(mapper类)而不仅仅是'Name'值。

2 个答案:

答案 0 :(得分:6)

如果您需要完整对象,则需要在子查询中按名称选择最大版本并加入到该对象中:

max_versions = session.query(Cls.name, func.max(Cls.version).label('max_version'))\
                      .group_by(Cls.name).subquery()
objs = session.query(Cls).join((max_versions,
           and_(Cls.name == max_versions.c.name,
                Cls.version == max_versions.c.max_version)
       )).all()

这将导致类似这样的事情:

SELECT tbl.id AS tbl_id, tbl.name AS tbl_name, tbl.version AS tbl_version
FROM tbl JOIN (SELECT tbl.name AS name, max(tbl.version) AS max_version
FROM tbl GROUP BY tbl.name) AS anon_1 ON tbl.name = anon_1.name AND tbl.version = anon_1.max_version

请注意,如果有多行具有最大版本,您将获得具有相同名称的多行。

答案 1 :(得分:-2)

以下是您可以使用engine.execute命令调用的SQL:

select
    t1.*
from
    table t1
    inner join 
        (select
            Name,
            max(version) as Version
         from
            table
         group by
            name) s on
        s.name = t1.name
        and s.version = t1.version