服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

Mysql|Sql Server|Oracle|Redis|MongoDB|PostgreSQL|Sqlite|DB2|mariadb|Access|数据库技术|

服务器之家 - 数据库 - Mysql - 探索MySQL递归查询:处理层次结构数据

探索MySQL递归查询:处理层次结构数据

2024-04-09 15:45数据库干货铺 Mysql

递归查询通过迭代处理分层数据的结果集来实现。在我们的案例中,初始查询选择了顶级领导,递归查询则利用较小层级结果,通过连接操作找到下一层级的员工,持续迭代直至到达最底层。递归查询每次迭代都使用前一次结果作

探索MySQL递归查询:处理层次结构数据

在数据库管理中,处理具有层次结构的数据一直是一项常见任务。MySQL的递归查询功能通过公用表表达式(CTE)为处理这类数据提供了便捷的方式。递归查询可以用于管理组织结构、目录树等数据,使您能够轻松地查询任意节点的子节点、父节点或整个路径。

1. 语法解释

在MySQL中,递归查询的基本语法结构如下所示:

WITH RECURSIVE cte_name AS (
    -- 初始查询(第一次迭代)
    SELECT initial_query
    UNION ALL
    -- 递归查询(后续迭代)
    SELECT recursive_query
    FROM cte_name
    JOIN base_table ON join_condition
)
-- 最终查询
SELECT * FROM cte_name;

在这个语法中,cte_name 是公用表表达式的名称,initial_query 是初始查询,recursive_query 是递归查询部分,base_table 是要进行递归的基本表,join_condition 是连接条件。

2.  案例演示

下面通过一个实际案例来展示如何在MySQL中利用递归查询处理组织结构数据。假设我们有一个名为employees的表,包含员工的id、姓名和直接上级的id。我们的目标是查询每个员工的直接上级、上级的上级,一直到顶级领导的完整路径。演示的环境为MySQL8.0环境。

CREATE TABLE employees (
    id INT,
    name VARCHAR(50),
    manager_id INT
);


INSERT INTO employees VALUES
(1, 'Alice', NULL),
(2, 'Bob', 1),
(3, 'Charlie', 2),
(4, 'David', 2),
(5, 'Eve', 1);

现在,让我们使用递归查询来获得每个员工的完整上级路径:

WITH RECURSIVE emp_path AS (
    SELECT id, name, 1 as level, CAST(name AS CHAR(200)) as path
    FROM employees
    WHERE manager_id IS NULL
    UNION ALL
    SELECT e.id, e.name, ep.level + 1, CONCAT(ep.path, ' -> ', e.name)
    FROM employees e
    JOIN emp_path ep ON e.manager_id = ep.id
)
SELECT * FROM emp_path;

查询结果如下:

探索MySQL递归查询:处理层次结构数据

3.  MySQL5.7中的实现

在 MySQL 5.7 中,递归查询不支持使用公用表表达式(CTE),而是通过使用用户定义变量(User-Defined Variables)和自连接(Self Join)来实现。虽然这种方法比较繁琐,但仍然可以实现递归查询。例如:
SELECT 
    t1.id as emp_id,
    t1.name as emp_name,
    t1.manager_id as manager_id,
    t1.name as emp_path,
    @pv := t1.manager_id as 'parent_id',
    @path := t1.name as 'path'
FROM 
    employees t1
JOIN 
    (SELECT @pv := '2', @path := '') tmp
WHERE t1.id = @pv


UNION


SELECT 
    t2.id as emp_id,
    t2.name as emp_name,
    t2.manager_id as manager_id,
    CONCAT(@path, ' -> ', t2.name) as emp_path,
    @pv := t2.manager_id as 'parent_id',
    @path := CONCAT(@path, ' -> ', t2.name) as 'path'
FROM 
    employees t2
JOIN 
    (SELECT @pv, @path) tmp
WHERE t2.id = @pv

查询结果如下:

探索MySQL递归查询:处理层次结构数据

 这个查询通过使用用户定义变量 @pv 和 @path 来保存父级的 ID 和路径,然后通过自连接不断迭代地找到每个员工的直接上级以及完整的上级路径。注意这是一种近似的实现,可能不如 CTE 那样直观和简洁。

当然如果需求比较简单的递归也可以用其他方式实现,具体看表设计情况及数据层级关系而编写脚本。

4. 递归查询原理与使用场景

递归查询通过迭代处理分层数据的结果集来实现。在我们的案例中,初始查询选择了顶级领导,递归查询则利用较小层级结果,通过连接操作找到下一层级的员工,持续迭代直至到达最底层。递归查询每次迭代都使用前一次结果作为输入,从而构建完整的层级关系。

递归查询的关键在于设计良好的初始查询和递归查询部分,以确保每次迭代都能准确找到下一层数据并连接到前一次的结果。

通过递归查询,可以轻松处理树形数据结构,解决组织结构、目录树等具有分层关系的数据问题,为数据分析提供了便利。

递归查询在实际应用中还能快速准确地分析和查找复杂层级数据关系,提升数据处理效率和准确性。

希望这篇文章能帮助您了解MySQL中的递归查询,以及如何利用这一功能处理层次结构数据。

原文地址:https://mp.weixin.qq.com/s/EK5K_hMoL908hhY1T1Cagg

延伸 · 阅读

精彩推荐
  • Mysql使用MySQL实现select into临时表的功能

    使用MySQL实现select into临时表的功能

    这篇文章主要介绍了使用MySQL实现select into临时表的功能,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...

    三爷麋了鹿8882022-11-16
  • MysqlMySQL启动错误解决方法

    MySQL启动错误解决方法

    本文给大家分享的是mysql启动时报错的排查过程及方法,非常实用,有相同问题的小伙伴可以来参考下...

    yayun5042020-07-04
  • MysqlMySQL存储全角字符和半角字符的区别

    MySQL存储全角字符和半角字符的区别

    这篇文章主要介绍了MySQL存储全角字符和半角字符的区别的相关资料,需要的朋友可以参考下...

    linux_c_coding_man5572020-07-29
  • MysqlMySQL中IO问题的深入分析与优化

    MySQL中IO问题的深入分析与优化

    据库作为存储系统,所有业务访问数据的操作都会转化为底层数据库系统的IO行为,下面这篇文章主要给大家介绍了关于MySQL中IO问题的深入分析与优化的相关...

    王世员7642022-10-14
  • Mysqllinux CentOS6.5 yum安装mysql5.6

    linux CentOS6.5 yum安装mysql5.6

    这篇文章主要为大家详细介绍了linux CentOS6.5 yum安装mysql5.6的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下 ...

    别踩我袈裟7622020-07-31
  • Mysqljava连不上mysql8.0问题的解决方法

    java连不上mysql8.0问题的解决方法

    这篇文章主要为大家详细介绍了java连不上MySQL8.0问题的解决方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    cungudafa7052020-10-07
  • MysqlMYSQL 批量替换之replace语法的使用详解

    MYSQL 批量替换之replace语法的使用详解

    本篇文章是对MYSQL中replace语法的使用进行了详细的分析介绍,需要的朋友参考下 ...

    MYSQL教程网3502020-01-05
  • Mysqlmysql索引原理与用法实例分析

    mysql索引原理与用法实例分析

    这篇文章主要介绍了mysql索引原理与用法,结合实例形式分析了mysql索引的基本概念、原理、用法及操作注意事项,需要的朋友可以参考下...

    随风行云6372021-01-14