Mysql 查询语句实现无限层次父子关系查询

目录

一、前言

二、相关语法函数介绍

三、具体实现

四、效率问题


一、前言

单表存储上下级关系,使用mysql 内置函数循环递归查出来


二、相关语法函数介绍

@

 @是用户变量,@@是系统变量。

:= 

不只在set和update时时赋值的作用,在select也是赋值的作用。

group_concat()  

将group by产生的同一个分组中的值连接起来,返回一个字符串结果。

FIND_IN_SET()

查询字段(strlist)中包含(str)的结果,返回结果为null或记录


三、具体实现

创建表


查询父级为 2 的下级  无限级查询

	SELECT
		@ids AS _ids,
		( SELECT @ids := GROUP_CONCAT( descendant ) FROM relation WHERE FIND_IN_SET( ancestor, @ids ) ) AS cids,
		@l := @l + 1 AS LEVEL 
	FROM
		relation,
		( SELECT @ids := 2, @l := 0 ) b 
	WHERE
		@ids IS NOT NULL 

列表化 

SELECT
	descendant.LEVEL,
	DATA.* 
FROM
	(
	SELECT
		@ids AS _ids,
		( SELECT @ids := GROUP_CONCAT( descendant ) FROM relation WHERE FIND_IN_SET( ancestor, @ids ) ) AS cids,
		@l := @l + 1 AS LEVEL 
	FROM
		relation,
		( SELECT @ids := 2, @l := 0 ) b 
	WHERE
		@ids IS NOT NULL 
		) descendant,
	sys_dept_relation DATA 
WHERE
	FIND_IN_SET( DATA.descendant, descendant._ids ) 
ORDER BY
	LEVEL,
	DATA.descendant


 查询ID为 8 的上级  无限级查询

	SELECT
		@id AS _id,
		( SELECT @id := ancestor FROM relation WHERE descendant = @id ) AS _pid,
		@l := @l + 1 AS LEVEL 
	FROM
		relation,
		( SELECT @id := 8, @l := 0 ) b 
	WHERE
		@id > 0 

四、效率问题

我目前测试表有8千多数据量

 


EXPLAN 相关参数

Select_type:

PRIMARY:查询中最外层的SELECT(如两表做UNION或者存在子查询的外层的表操作为PRIMARY,内层的操作为UNION)

DERIVED:被驱动的SELECT子查询(子查询位于FROM子句)

UNCACHEABLE SUBQUERY一个子查询的结果不能被缓存,必须重新评估外链接的第一行

Type:

NULL: MySQL在优化过程中分解语句,执行时甚至不用访问表或索引,例如从一个索引列里选取最小值可以通过单独索引查找完成。

index: Full Index Scan,index与ALL区别为index类型只遍历索引树

System:system是const类型的特例,当查询的表只有一行的情况下,使用system

 

 

  • 6
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 回答1: MySQL 递归查询语言主要是通过使用 `WITH RECURSIVE` 关键字来实现的,类似于其他数据库管理系统中的 `WITH RECURSIVE` 关键字。 下面是一个使用 MySQL 递归查询语言的例子,假设我们有一个格 `employee` 包含员工的 ID、姓名和上 ID: ```sql CREATE TABLE `employee` ( `id` INT NOT NULL, `name` VARCHAR(45) NOT NULL, `manager_id` INT NULL, PRIMARY KEY (`id`) ); INSERT INTO `employee` (`id`, `name`, `manager_id`) VALUES (1, 'Alice', NULL), (2, 'Bob', 1), (3, 'Charlie', 2), (4, 'David', 2), (5, 'Eve', 1), (6, 'Frank', 5), (7, 'Grace', 5), (8, 'Henry', 7); ``` 现在我们想要查询每个员工的直接上以及所有上的姓名,可以使用递归查询语言来实现: ```sql WITH RECURSIVE cte (id, name, manager_id, level) AS ( SELECT id, name, manager_id, 0 FROM employee WHERE id = 6 -- 员工 ID UNION ALL SELECT e.id, e.name, e.manager_id, c.level + 1 FROM employee e JOIN cte c ON e.id = c.manager_id ) SELECT CONCAT(REPEAT(' ', level), name) AS name, level FROM cte ORDER BY level; ``` 这个查询语句会输出以下结果: ``` Frank 0 Eve 1 Grace 2 Henry 3 Alice 1 ``` 这个结果示 Frank 的直接上是 Eve,Eve 的直接上是 Alice,而 Grace 和 Henry 是 Eve 的上,Alice 没有。 ### 回答2: MySQL递归查询语言是指在MySQL数据库中使用递归查询实现层次结构数据的查询递归查询可以用于处理带有父子关系的数据,例如组织机构、分类目录等。 MySQL递归查询语言的实现方式是通过使用WITH RECURSIVE子句来定义递归查询。这个子句包含两部分:递归部分和终止条件部分。递归部分定义了每一次递归查询时如何从上一次的结果中获取下一的数据,而终止条件部分定义了递归查询何时停止。 在递归查询中,通过使用UNION操作符将递归部分和终止条件部分连接在一起。递归部分的查询语句中使用了上一次递归查询的结果作为输入,并且在结果中筛选出下一的数据。终止条件部分的查询语句用于获取最顶层的数据。 递归查询语言的基本语如下: ``` WITH RECURSIVE cte_name (column1, column2, ...) AS ( SELECT initial_query UNION ALL SELECT recursive_query ) SELECT * FROM cte_name; ``` 递归查询语言的应用场景很多,例如可以用来查询组织机构层级结构、获取分类目录的嵌套关系等。通过递归查询,可以轻松地遍历和操作具有层次结构的数据。 ### 回答3: MySQL没有专门的递归查询语言。但是可以通过使用存储过程和临时实现递归查询。 在MySQL中,可以通过编写存储过程来实现递归查询。存储过程是一种预先编译的SQL语句集合,可以在MySQL数据库上进行执行。在存储过程中,可以使用循环和条件语句来进行递归查询。通过逐步迭代和条件判断,可以实现对具有层级结构的数据进行递归查询。 此外,临时也是实现递归查询的一种方。临时是一种临时存储数据的,它只存在于当前会话中,并在会话结束后自动被删除。通过创建多个临时和通过循环插入数据到临时,可以模拟递归查询的过程。 总的来说,虽然MySQL没有原生的递归查询语言,但是可以通过存储过程和临时等方实现递归查询递归查询在处理具有层级结构的数据时非常有用,可以实现对树形结构数据的遍历和查询。但是需要注意的是,递归查询可能会导致性能问题,所以在使用时要谨慎考虑。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值