11

这是我在 PostgreSQL 中遇到的问题的简化版本。

我有下表A

[ ID 整数 | 数值 数字(10,2) | 父 整数 ]

其中 'PARENT' 是对列 ID 的自引用 FK。

表定义为:

CREATE TABLE A(ID INTEGER IDENTITY, VALUE NUMERIC(10,2), PARENT INTEGER)                                    
ALTER  TABLE A ADD CONSTRAINT FK FOREIGN KEY (PARENT) REFERENCES A(ID) 

这个简单的表允许定义任意深度的树数据结构。现在我需要编写一个 SQL(我不喜欢使用服务器端 PL-SQL)来报告每个节点,它下面的子树“挂起”的总值。例如,使用下表:

|  ID  | VALUE | PARENT |
-------------------------
|   1  | NULL  | NULL   |
|   2  | 3.50  |    1   |
|   3  | NULL  | NULL   |
|   4  | NULL  |    3   |
|   5  | 1.50  |    4   |
|   6  | 2.20  |    4   |

我应该得到以下结果集:

| ID  |  Total-Value-of-Subtree |
|  1  |                  3.50   |
|  2  |                  3.50   |
|  3  |                  3.70   |
|  4  |                  3.70   |
|  5  |                  1.50   |
|  6  |                  2.20   |

简单来说,您可以假设只有叶节点有值,非叶节点在VALUE列中始终具有NULL值。有没有办法在 SQL 中做到这一点,甚至使用 PostgreSQL 特定的扩展?

4

2 回答 2

7

在 PostgreSQL 中,您可以使用递归 CTE(通用表表达式)在查询中遍历树。

以下是文档的两个相关链接:

编辑

由于不需要子选择,它在更大的数据集上可能比 Arion 的查询运行得更好。

WITH RECURSIVE children AS (
    -- select leaf nodes
    SELECT id, value, parent
        FROM t
        WHERE value IS NOT NULL
    UNION ALL
    -- propagate values of leaf nodes up, adding rows 
    SELECT t.id, children.value, t.parent
        FROM children JOIN t ON children.parent = t.id
)
SELECT id, sum(value) 
    FROM children 
    GROUP BY id   -- sum up appropriate rows
    ORDER BY id;
于 2012-11-02T09:04:10.013 回答
4

也许是这样的:

WITH RECURSIVE CTE
AS
(
    SELECT
        t.ID,
        t.VALUE,
        t.PARENT
    FROM
        t
    WHERE NOT EXISTS
        (
            SELECT NULL FROM t AS t2 WHERE t2.PARENT=t.ID
        )
    UNION ALL
    SELECT
        t.ID,
        COALESCE(t.VALUE,CTE.VALUE),
        t.PARENT
    FROM
        t
        JOIN CTE
            ON CTE.PARENT=t.ID
)
SELECT
    CTE.ID,
    SUM(CTE.VALUE)
FROM
    CTE
GROUP BY
    CTE.ID
ORDER BY 
    ID;

这将从没有孩子的孩子开始。然后上树去找父母。结果将是这样的:

1   3.50
2   3.50
3   3.70
4   3.70
5   1.50
6   2.20
于 2012-11-02T09:02:56.233 回答