加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

寻找生成森林(WITH RECURSIVE,PostgreSQL 9.5)

发布时间:2020-12-13 15:53:06 所属栏目:百科 来源:网络整理
导读:我有一个任意数量的人的身份表(即别名).每行都有一个以前的名称和一个新名称.在生产中,大约有1M行.例如: id,old,new---1,'Albert','Bob'2,'Bob','Charles'3,'Mary','Nancy'4,'Charles','Albert'5,'Lydia','Nancy'6,'Zoe','Zoe' 我想要的是生成用户列表并引
我有一个任意数量的人的身份表(即别名).每行都有一个以前的名称和一个新名称.在生产中,大约有1M行.例如:

id,old,new
---
1,'Albert','Bob'
2,'Bob','Charles'
3,'Mary','Nancy'
4,'Charles','Albert'
5,'Lydia','Nancy'
6,'Zoe','Zoe'

我想要的是生成用户列表并引用他们各自的身份.这类似于查找连接标识的每个图形中的所有节点,或查找生成林:

User 1: Albert,Bob,Charles (identities: 1,2,4)
User 2: Mary,Nancy,Lydia (identities: 3,5)
User 3: Zoe (identities: 6)

我一直在修补PostgreSQL的WITH RECURSIVE,但它为每个产生了集合和子集.例如:

1,4 <-- spanning tree: good
2     <-- subset: discard
3,5   <-- spanning tree: good
4     <-- subset: discard
5     <-- subset: discard
6     <-- spanning tree: good

我需要做什么才能为每个用户生成完整的身份集(即生成树)?

SQLFiddle:http://sqlfiddle.com/#!15/9eaed/4我的最新尝试.这是代码:

WITH RECURSIVE search_graph AS (
   SELECT id,id AS min_id,ARRAY[id] AS path,ARRAY[old,new] AS emails
   FROM   identities

   UNION 

   SELECT identities.id,LEAST(identities.id,sg.min_id),(sg.path || identities.id),(sg.emails || identities.old || identities.new)

   FROM search_graph sg
   JOIN identities ON (identities.old = ANY(sg.emails) OR identities.new = ANY(sg.emails))
   WHERE  identities.id <> ALL(sg.path)
)
SELECT array_agg(DISTINCT(p)) from search_graph,unnest(path) p GROUP BY min_id;

结果如下:

1,4
2
3,5
4
5
6

解决方法

我刚才写了一个类似问题的答案: How to find all connected subgraphs of an undirected graph.在那个问题中,我使用了SQL Server.有关中间CTE的详细说明,请参阅该答案.我将该查询改编为Postgres.

使用Postgres数组功能可以更有效地编写它,而不是将路径连接到文本列.

WITH RECURSIVE
CTE_Idents
AS
(
    SELECT old AS Ident
    FROM identities

    UNION

    SELECT new AS Ident
    FROM identities
),CTE_Pairs
AS
(
    SELECT old AS Ident1,new AS Ident2
    FROM identities
    WHERE old <> new

    UNION

    SELECT new AS Ident1,old AS Ident2
    FROM identities
    WHERE old <> new
),CTE_Recursive
AS
(
    SELECT
        CTE_Idents.Ident AS AnchorIdent,Ident1,Ident2,',' || Ident1 || ',' || Ident2 || ',' AS IdentPath,1 AS Lvl
    FROM 
        CTE_Pairs
        INNER JOIN CTE_Idents ON CTE_Idents.Ident = CTE_Pairs.Ident1

    UNION ALL

    SELECT 
        CTE_Recursive.AnchorIdent,CTE_Pairs.Ident1,CTE_Pairs.Ident2,CTE_Recursive.IdentPath || CTE_Pairs.Ident2 || ',CTE_Recursive.Lvl + 1 AS Lvl
    FROM
        CTE_Pairs
        INNER JOIN CTE_Recursive ON CTE_Recursive.Ident2 = CTE_Pairs.Ident1
    WHERE
        CTE_Recursive.IdentPath NOT LIKE ('%,' || CTE_Pairs.Ident2 || ',%')
),CTE_RecursionResult
AS
(
    SELECT AnchorIdent,Ident2
    FROM CTE_Recursive
),CTE_CleanResult
AS
(
    SELECT AnchorIdent,Ident1 AS Ident
    FROM CTE_RecursionResult

    UNION

    SELECT AnchorIdent,Ident2 AS Ident
    FROM CTE_RecursionResult
),CTE_Groups
AS
(
  SELECT
    CTE_Idents.Ident,array_agg(COALESCE(CTE_CleanResult.Ident,CTE_Idents.Ident) 
        ORDER BY COALESCE(CTE_CleanResult.Ident,CTE_Idents.Ident)) AS AllIdents
  FROM
    CTE_Idents
    LEFT JOIN CTE_CleanResult ON CTE_CleanResult.AnchorIdent = CTE_Idents.Ident
  GROUP BY CTE_Idents.Ident
)
SELECT AllIdents
FROM CTE_Groups
GROUP BY AllIdents
;

我在样本数据中添加了一行(7,X,Y).

SQL Fiddle

结果

|          allidents |
|--------------------|
|   Lydia,Mary,Nancy |
| Albert,Charles |
|                X,Y |
|                Zoe |

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读