0

以下是我拥有的表格格式。表名 :: USERS

userid reporttouserid
------ ------------
101    NULL
102    101
103    102

现在我需要一个查询来列出 101 下的所有子用户 ID,即 102 和 103(103 间接低于 101,因为其父级 102 低于 101)

我已经在 postgresql 中看到了公共表表达式,但无法弄清楚如何去做。

4

1 回答 1

5

PostgreSQL 文档涵盖了这个主题。请参阅该页面上给出的递归 CTE 示例。

递归 CTE 可能有点难以掌握,但一旦使用它们就会非常强大。阅读文档并进行一些实验;你会明白的。

(请始终提及您的 PostgreSQL 版本,并在您的问题中以表格形式显示所需的输出)。

给定演示数据:

create table users (
  userid integer primary key,
  reporttouserid integer references users(userid)
);

insert into users(userid, reporttouserid) values (101,null), (102,101), (103,102);

(请在可能的问题中提供这个,创建它很痛苦)

您可以使用以下方法递归地遍历图形:

WITH RECURSIVE flatusers(userid, reporttouserid, baseuserid) AS (
    SELECT userid, reporttouserid, userid AS baseuserid 
    FROM users WHERE reporttouserid IS NULL
    UNION ALL
    SELECT u.userid, u.reporttouserid, f.baseuserid
    FROM flatusers f
    INNER JOIN users u ON f.userid = u.reporttouserid

)
SELECT * FROM flatusers;

产生如下输出:

 userid | reporttouserid | baseuserid 
--------+----------------+------------
    101 |                |        101
    102 |            101 |        101
    103 |            102 |        101
(3 rows)

我相信你可以弄清楚从那里去哪里。在使用它之前,请确保您了解递归 CTE。

请注意,PostgreSQL(至少 9.4 或更早版本)不能(不幸地)将 quals 下推到 CTE 术语中,即使对于非递归 CTE 也是如此。如果添加WHERE baseuserid = 101到查询中,查询仍会生成整个展平表,然后将其大部分丢弃。如果您只想对一个baseuserid 执行此递归操作,则必须在递归 CTE 术语的静态联合部分WHERE之后添加适当的子句术语。WHERE reporttouserid IS NULL

于 2013-11-12T00:53:06.440 回答