Oracle 数据库中递归查询

方法一:使用 CONNECT BY(传统方式)

-- 假设表名为 my_table,字段为 id 和 parent_id

SELECT id

FROM my_table

START WITH id = '根节点ID' -- 从根节点开始

CONNECT BY PRIOR id = parent_id -- 递归条件:父节点的 id = 子节点的 parent_id

MINUS -- 去重(可选)

-- 或者使用 NOT EXISTS 来筛选叶子节点

SELECT id

FROM my_table t

WHERE NOT EXISTS (

SELECT 1

FROM my_table t2

WHERE t2.parent_id = t.id

)

START WITH id = '根节点ID'

CONNECT BY PRIOR id = parent_id;

方法二:使用 WITH 递归(推荐,更标准)

WITH RECURSIVE cte AS (

-- 基础查询:从根节点开始

SELECT id, parent_id, 1 as level

FROM my_table

WHERE id = '根节点ID' -- 替换为实际的根节点ID

UNION ALL

-- 递归查询:查找所有子节点

SELECT t.id, t.parent_id, c.level + 1

FROM my_table t

INNER JOIN cte c ON t.parent_id = c.id

)

-- 筛选叶子节点(没有子节点的节点)

SELECT c.id

FROM cte c

WHERE NOT EXISTS (

SELECT 1

FROM my_table t

WHERE t.parent_id = c.id

);

方法三:一步到位(最简洁)

WITH cte AS (

SELECT id, parent_id

FROM my_table

START WITH id = '根节点ID'

CONNECT BY PRIOR id = parent_id

)

SELECT id

FROM cte c

WHERE NOT EXISTS (

SELECT 1 FROM my_table WHERE parent_id = c.id

);

示例

假设数据如下:

id | parent_id

----|----------

A | NULL (根)

B | A

C | A

D | B

E | B

查询根节点 A 的所有叶子节点:

SELECT id

FROM my_table

START WITH id = 'A'

CONNECT BY PRIOR id = parent_id

MINUS

SELECT id

FROM my_table

WHERE parent_id IS NOT NULL;

结果:C, D, E

性能优化建议

  1. 创建索引:

CREATE INDEX idx_parent_id ON my_table(parent_id);

CREATE INDEX idx_id ON my_table(id);

  1. 如果数据量大,考虑使用 NO_CYCLE:

START WITH id = '根节点ID'

CONNECT BY NOCYCLE PRIOR id = parent_id

相关推荐
小陈工1 小时前
Python Web开发入门(十七):Vue.js与Python后端集成——让前后端真正“握手言和“
开发语言·前端·javascript·数据库·vue.js·人工智能·python
科技小花5 小时前
数据治理平台架构演进观察:AI原生设计如何重构企业数据管理范式
数据库·重构·架构·数据治理·ai-native·ai原生
一江寒逸5 小时前
零基础从入门到精通MySQL(中篇):进阶篇——吃透多表查询、事务核心与高级特性,搞定复杂业务SQL
数据库·sql·mysql
D4c-lovetrain5 小时前
linux个人心得22 (mysql)
数据库·mysql
阿里小阿希6 小时前
CentOS7 PostgreSQL 9.2 升级到 15 完整教程
数据库·postgresql
荒川之神6 小时前
Oracle 数据仓库雪花模型设计(完整实战方案)
数据库·数据仓库·oracle
做个文艺程序员6 小时前
MySQL安全加固十大硬核操作
数据库·mysql·安全
不吃香菜学java6 小时前
Redis简单应用
数据库·spring boot·tomcat·maven
一个天蝎座 白勺 程序猿7 小时前
Apache IoTDB(15):IoTDB查询写回(INTO子句)深度解析——从语法到实战的ETL全链路指南
数据库·apache·etl·iotdb
不知名的老吴7 小时前
Redis的延迟瓶颈:TCP栈开销无法避免
数据库·redis·缓存