postgresql递归查询指定搜索顺序的方法

原文地址

搜索顺序

在使用递归查询进行树遍历计算时,你可能希望按照深度优先或广度优先的顺序对结果进行排序。这可以通过在数据列之外再计算一个排序列来实现,并在最后使用该列对结果进行排序。需要注意的是,这实际上并不会控制查询评估访问行的顺序;这个顺序在 SQL 中始终依赖于具体实现。此方法仅提供了一种在事后对结果进行排序的便捷方式。

创建深度优先顺序

我们可以为每个结果行计算一个数组,用于记录截至目前已访问过的行。例如,考虑以下使用 link 字段搜索表 tree 的查询:

sql 复制代码
WITH RECURSIVE search_tree(id, link, data) AS (
    SELECT t.id, t.link, t.data
    FROM tree t
  UNION ALL
    SELECT t.id, t.link, t.data
    FROM tree t, search_tree st
    WHERE t.id = st.link
)
SELECT * FROM search_tree;

要添加深度优先排序信息,可以这样写:

sql 复制代码
WITH RECURSIVE search_tree(id, link, data, path) AS (
    SELECT t.id, t.link, t.data, ARRAY[t.id]
    FROM tree t
  UNION ALL
    SELECT t.id, t.link, t.data, path || t.id
    FROM tree t, search_tree st
    WHERE t.id = st.link
)
SELECT * FROM search_tree ORDER BY path;

在一般情况下,如果需要使用多个字段来标识一行,可以使用行数组。例如,如果需要追踪字段 f1f2

sql 复制代码
WITH RECURSIVE search_tree(id, link, data, path) AS (
    SELECT t.id, t.link, t.data, ARRAY[ROW(t.f1, t.f2)]
    FROM tree t
  UNION ALL
    SELECT t.id, t.link, t.data, path || ROW(t.f1, t.f2)
    FROM tree t, search_tree st
    WHERE t.id = st.link
)
SELECT * FROM search_tree ORDER BY path;

提示 :在常见的只需追踪一个字段的情况下,可以省略 ROW() 语法。这样可以使用简单数组而非复合类型数组,从而提高效率。

创建广度优先顺序

你可以添加一个列来追踪搜索的深度,例如:

sql 复制代码
WITH RECURSIVE search_tree(id, link, data, depth) AS (
    SELECT t.id, t.link, t.data, 0
    FROM tree t
  UNION ALL
    SELECT t.id, t.link, t.data, depth + 1
    FROM tree t, search_tree st
    WHERE t.id = st.link
)
SELECT * FROM search_tree ORDER BY depth;

为了获得稳定的排序,可以将数据列作为次要排序列添加。

提示:递归查询评估算法会以广度优先搜索的顺序产生输出。然而,这是一个实现细节,依赖它可能并不可靠。每一层内的行顺序肯定是未定义的,因此在任何情况下,可能都需要一些显式的排序。

内置语法

PostgreSQL 提供了内置语法来计算深度优先或广度优先的排序列。例如:

sql 复制代码
WITH RECURSIVE search_tree(id, link, data) AS (
    SELECT t.id, t.link, t.data
    FROM tree t
  UNION ALL
    SELECT t.id, t.link, t.data
    FROM tree t, search_tree st
    WHERE t.id = st.link
) SEARCH DEPTH FIRST BY id SET ordercol
SELECT * FROM search_tree ORDER BY ordercol;
sql 复制代码
WITH RECURSIVE search_tree(id, link, data) AS (
    SELECT t.id, t.link, t.data
    FROM tree t
  UNION ALL
    SELECT t.id, t.link, t.data
    FROM tree t, search_tree st
    WHERE t.id = st.link
) SEARCH BREADTH FIRST BY id SET ordercol
SELECT * FROM search_tree ORDER BY ordercol;

该语法在内部会被扩展为类似于上述手写形式的内容。SEARCH 子句指定了是需要深度优先还是广度优先搜索、用于排序的列列表,以及一个将包含可用于排序的结果数据的列名。该列将隐式添加到 CTE 的输出行中。

相关推荐
java1234_小锋2 小时前
Redis的热Key问题如何解决?
数据库·redis·缓存
wang6021252182 小时前
FastAPI框架为什么在启动时建表
数据库
男孩李2 小时前
linux下如何执行postgres数据库的sql文件
数据库·sql·postgresql
zwjapple3 小时前
MySQL SQL 面试核心考点与注意事项总结
数据库·sql·mysql
乐韵天城3 小时前
SpringBoot中如何手动开启数据库事务
数据库·spring boot
05大叔3 小时前
Spring Day02
数据库·sql·spring
默默前行的虫虫3 小时前
nicegui中多次调用数据库操作总结
数据库·python
鸽鸽程序猿3 小时前
【Redis】事务
数据库·redis·缓存
Knight_AL3 小时前
MySQL 分区表应用案例:优化数据管理与性能
数据库·mysql