在MySQL 8.0之前的版本中,查询递归数据结构通常需要借助存储过程循环遍历,或者多次自关联拼接结果,不仅代码复杂,执行效率也比较低。MySQL 8.0新增的WITH RECURSIVE公用表表达式功能,让递归查询变得简单直观,能够直接通过一条SQL语句获取完整的递归层级数据。

WITH RECURSIVE基本语法
WITH RECURSIVE的核心结构分为两部分:锚点成员和递归成员,整体语法格式如下:
WITH RECURSIVE cte_name (column1, column2, ...) AS (
-- 锚点成员:初始查询,返回递归的起点数据
SELECT initial_column1, initial_column2, ...
FROM table_name
WHERE 初始条件
UNION ALL
-- 递归成员:引用CTE自身,基于上一次的结果进行扩展查询
SELECT t.column1, t.column2, ...
FROM table_name t
INNER JOIN cte_name c ON t.parent_id = c.id
)
SELECT * FROM cte_name;
其中锚点成员只会执行一次,用来获取递归的起始数据,递归成员会反复执行,直到没有新的数据返回为止,两部分通过UNION ALL连接,注意不能使用UNION,否则会去重导致递归提前终止。
实际场景示例:查询组织架构层级
假设我们有一张员工表employee,存储公司的组织架构数据,表结构如下:
| 字段名 | 类型 | 说明 |
|---|---|---|
| id | INT | 员工ID,主键 |
| name | VARCHAR(50) | 员工姓名 |
| manager_id | INT | 上级领导ID,根节点的manager_id为NULL |
现在需要查询所有下属的层级关系,从CEO开始往下遍历所有员工,同时显示每个员工的层级深度,对应的查询语句如下:
WITH RECURSIVE org_cte (id, name, manager_id, level) AS (
-- 锚点成员:查询CEO,也就是manager_id为NULL的员工
SELECT id, name, manager_id, 0 AS level
FROM employee
WHERE manager_id IS NULL
UNION ALL
-- 递归成员:关联上级员工,获取下属,层级加1
SELECT e.id, e.name, e.manager_id, c.level + 1
FROM employee e
INNER JOIN org_cte c ON e.manager_id = c.id
)
SELECT id, name, manager_id, level FROM org_cte ORDER BY level, id;
执行上述语句后,会先返回CEO的信息,层级为0,然后依次返回CEO的直接下属,层级为1,再返回下属的下属,层级为2,直到所有员工都被查询出来。
递归查询的注意事项
- 递归成员中必须引用CTE自身的名称,否则就不是递归查询,会直接报错。
- 要避免出现无限递归的情况,比如数据中存在父子关系的闭环,递归会一直执行直到超过MySQL的递归深度限制,默认递归深度是1000,可以通过调整
cte_max_recursion_depth参数修改,但是不建议设置过大,最好先检查数据的合理性。 - 锚点成员和递归成员的列数量必须一致,对应的列数据类型也要兼容,否则查询会执行失败。
- 如果需要对递归结果去重,可以在最外层的查询中使用
DISTINCT,不要在递归成员内部使用UNION去重,否则会干扰递归的执行逻辑。
向上递归查询场景
除了向下遍历查询所有下属,也可以使用WITH RECURSIVE向上查询某个员工的所有上级领导,比如查询id为10的员工的所有上级,语句如下:
WITH RECURSIVE upward_cte (id, name, manager_id) AS (
-- 锚点成员:查询当前员工
SELECT id, name, manager_id
FROM employee
WHERE id = 10
UNION ALL
-- 递归成员:关联上级领导
SELECT e.id, e.name, e.manager_id
FROM employee e
INNER JOIN upward_cte c ON e.id = c.manager_id
)
SELECT id, name, manager_id FROM upward_cte;
这个查询会从id为10的员工开始,依次向上查找其上级、上级的上级,直到根节点的CEO,完整返回整个上级链路。
MySQL_8.0WITH_RECURSIVE递归查询公用表表达式修改时间:2026-06-25 13:24:31