以下是我拥有的表格格式。表名 :: USERS
userid reporttouserid
------ ------------
101 NULL
102 101
103 102
现在我需要一个查询来列出 101 下的所有子用户 ID,即 102 和 103(103 间接低于 101,因为其父级 102 低于 101)
我已经在 postgresql 中看到了公共表表达式,但无法弄清楚如何去做。
以下是我拥有的表格格式。表名 :: USERS
userid reporttouserid
------ ------------
101 NULL
102 101
103 102
现在我需要一个查询来列出 101 下的所有子用户 ID,即 102 和 103(103 间接低于 101,因为其父级 102 低于 101)
我已经在 postgresql 中看到了公共表表达式,但无法弄清楚如何去做。
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