Mysql树形表的2种查询解决方案(递归与自连接)

 更新时间:2023年11月25日 08:35:04   作者:懒羊羊.java  
MySQL作为一个关系型数据库,存储着许多的数据信息,在实际应用中经常会遇到需要存储树形结构数据的情境,例如部门结构、商品分类等,这篇文章主要给大家介绍了关于Mysql树形表的2种查询解决方案,分别是递归与自连接,需要的朋友可以参考下

你有没有遇到过这样一种情况:

一张表就实现了一对多的关系,并且表中每一行数据都存在“爷爷-父亲-儿子-…”的联系,这也就是所谓的树形结构

对于这样的表很显然想要通过查询来实现价值绝对是不能只靠select * from table 来实现的,下面提供两种解决方案:

1.自连接

inner join 关键可以实现多种分类的查询,其实SQL很简单

SELECT
	one.id one_id,
	one.label one_label,
	two.id two_id,
	two.label two_label
FROM
	course_category one
	INNER JOIN course_category two ON two.parentid=one.id
	INNER JOIN course_category three ON three.parentid=two.id
	WHERE one.id='1' AND one.is_show='1' AND two.is_show='1'
	ORDER BY one.orderby,two.orderby

也是规规矩矩的就查出一整棵树

这种查询的原则就是通过parentId去实现,“爷爷找爸爸,爸爸找儿子,儿子找孙子”,下面来逐帧慢放:

1.one

2.one,two

3.one,two,three

可以看到,只有在树的层级确定的情况下我才能选择性的去自连接子表,某种意义上来讲这种方法存在弊端,我要是insert进去层级更低的新子节点那我的sql就得改变,从而就造成了一个“动一发而牵全身”的硬编码问题,实在是不够稳妥!

2.递归!

向上递归

首先声明,如果mysql的版本低于8是不支持递归查询的函数的!

下面来看一下如何用递归优雅的实现,从树根查到树顶:

先来看一个简单的Demo

	with RECURSIVE t1 AS(
		SELECT 1 AS n
		union all
		SELECT n+1 FROM t1 WHERE n<5
	)
	SELECT * from t1

该怎么理解这每一步呢?
 

WITH RECURSIVE t1 AS:

这是递归查询的开始,创建了一个名为t1的递归表。

SELECT 1 AS n:

在t1表中,插入了一个初始行,值为1,命名为n。

UNION ALL:

使用UNION ALL运算符将初始行和递归查询结果合并,形成递归步骤。这也就是下次递归的起点表

SELECT n+1 FROM t1 WHERE n<5:

递归部分的查询,从t1表中选择n加1的结果,当n小于5时进行递归。

SELECT * FROM t1:

最终查询,返回t1表的所有行。

其实在使用递归的过程只需要注意要去避免死龟就好!

如何去查开头的那张树形表呢?这样就好:

with recursive temp as (
select * from  course_category p where  id= '1'
 union all
select t.* from course_category t inner join temp on temp.id = t.parentid
)
select *  from temp order by temp.id, temp.orderby

下面我们逐帧分析:

其实关键的地方就在于第三步,在树根的基础上去找叶子:

神之一手:
select t.* from course_category t inner join temp on temp.id = t.parentid
这就是递归相较于第一种方式可以无视层级inner jion的关键,因为这个动作已经被递归自动完成了,递归巧妙地一点就在这里!

向下递归

基于向上递归父找子的思想,向下递归则是子找父,即在叶子基础上union all之后去找根

子的parentId=父的id

with recursive temp as (
select * from  course_category p where  id= '1-1-1'
 union all
select t.* from course_category t inner join temp on temp.parentid = t.id  
//temp表是下次递归的基础
)
select *  from temp order by temp.id, temp.orderby

值得注意的是Mysql为了避免无限递归递归次数为1000次,也可以人为来设置cte_max_recursion_depth和max_execution_time来自定义递归深度和执行时间

使用递归的好处无需言语,一次io连接就搞定了全部

总结

到此这篇关于Mysql树形表的2种查询解决方案的文章就介绍到这了,更多相关Mysql树形表查询内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • MySQL与PHP的基础与应用专题之索引

    MySQL与PHP的基础与应用专题之索引

    MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,本系列将带你掌握php与mysql的基础应用,本篇从索引开始
    2022-02-02
  • 关于 MySQL 嵌套子查询中无法关联主表字段问题的解决方法

    关于 MySQL 嵌套子查询中无法关联主表字段问题的解决方法

    这篇文章主要介绍了关于 MySQL 嵌套子查询中,无法关联主表字段问题的折中解决方法,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-12-12
  • Mysql中日期和时间函数应用不用求人

    Mysql中日期和时间函数应用不用求人

    Mysql中日期和时间函数应用不用求人,学习mysql的朋友可以参考下。
    2010-11-11
  • mysql列转行以及年月分组实例

    mysql列转行以及年月分组实例

    下面小编就为大家带来一篇mysql列转行以及年月分组实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-03-03
  • 为MySQL安装配置代理工具Kingshard的基本教程

    为MySQL安装配置代理工具Kingshard的基本教程

    这篇文章主要介绍了为MySQL安装配置代理工具Kingshard的基本教程,Kingshard由Go语言写成,可以实现读写分离和客户端IP访问控制等功能,非常强大,需要的朋友可以参考下
    2015-12-12
  • ssm框架如何调用mysql存储过程

    ssm框架如何调用mysql存储过程

    这篇文章主要介绍了ssm框架如何调用mysql存储过程,首先是建表,创建存储过程,本文结合示例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧
    2024-05-05
  • mysql中inner join和left join使用详解

    mysql中inner join和left join使用详解

    本文主要介绍了mysql中inner join和left join使用详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-03-03
  • mysql的日期和时间函数

    mysql的日期和时间函数

    mysql的日期和时间函数 这里是一个使用日期函数的例子。
    2010-11-11
  • MySQL派生表联表查询实战过程

    MySQL派生表联表查询实战过程

    派生表是查询结果组成的虚拟表,派生表是在外部查询的FROM子句中定义的,不需要手动创建,下面这篇文章主要给大家介绍了关于MySQL派生表联表查询的相关资料,需要的朋友可以参考下
    2022-03-03
  • mysql alter table修改表命令整理

    mysql alter table修改表命令整理

    这篇文章主要介绍了mysql alter table修改表命令整理的相关资料,需要的朋友可以参考下
    2016-10-10

最新评论