mysql通过@变量实现递归详细实例

 更新时间:2023年06月07日 10:34:57   作者:supuerlovepc  
众所周知目前的mysql版本中并不支持直接的递归查询,下面这篇文章主要给大家介绍了关于mysql通过@变量实现递归的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下

1. 测试表结构

create table demo
(
    id varchar(100) not null
        primary key,
    parentId    varchar(100) not null
);
-- 测试数据
INSERT INTO demo (ID, parentID) VALUES ('1', '0');
INSERT INTO demo (ID, parentID) VALUES ('10', '9');
INSERT INTO demo (ID, parentID) VALUES ('11', '9');
INSERT INTO demo (ID, parentID) VALUES ('2', '1');
INSERT INTO demo (ID, parentID) VALUES ('3', '2');
INSERT INTO demo (ID, parentID) VALUES ('4', '3');
INSERT INTO demo (ID, parentID) VALUES ('5', '4');
INSERT INTO demo (ID, parentID) VALUES ('6', '5');
INSERT INTO demo (ID, parentID) VALUES ('7', '6');
INSERT INTO demo (ID, parentID) VALUES ('8', '1');
INSERT INTO demo (ID, parentID) VALUES ('9', '1');

2. 向下递归(包括自己)

-- 向下递归 par即为递归查询出来的所有id
SELECT
    @par                                                                          AS par,
    (SELECT @par := GROUP_CONCAT(id) FROM demo WHERE FIND_IN_SET(parentid, @par)) AS son
FROM demo_userm, (SELECT @par := '1') T
WHERE @par IS NOT NULL;
查询结果:
| par       | son     |
| 1         | 2,8,9   |
| 2,8,9     | 10,11,3 |
| 10 ,11,3  | 4       |
| 4         | 5       |
| 5         | 6       |
| 6         | 7       |
| 7         | NULL    |

3.向下递归(不包括自己)

-- 向下递归 par即为递归查询出来的所有id
SELECT
    @par                                                                          AS par,
    (SELECT @par := GROUP_CONCAT(id) FROM demo WHERE FIND_IN_SET(parentid, @par)) AS son
FROM demo_userm, (SELECT @par := '1') T
WHERE @par IS NOT NULL and @par != '1';
查询结果:
| par       | son     |
| 2,8,9     | 10,11,3 |
| 10 ,11,3  | 4       |
| 4         | 5       |
| 5         | 6       |
| 6         | 7       |
| 7         | NULL    |

4. 向下递归可能存在问题,修正优化:

-- 如果存在 par = 51016 并且 son = 51016 , 那么会出现问题,多出许多数据
SELECT
    @par                                                                          AS par,
    (SELECT @par := GROUP_CONCAT(distinct ORG_REFNO) FROM gis_udp_bank WHERE FIND_IN_SET(PNBRN_ORG_REFNO, @par)) AS son
FROM gis_udp_bank, (SELECT @par := '51016') T
WHERE @par IS NOT NULL;
-- 优化后:后面家条件: !FIND_IN_SET(ORG_REFNO, @par)
SELECT
    @par                                                                          AS par,
    (SELECT @par := GROUP_CONCAT(distinct ORG_REFNO) FROM gis_udp_bank WHERE FIND_IN_SET(PNBRN_ORG_REFNO, @par)
        and !FIND_IN_SET(ORG_REFNO, @par)
        ) AS son
FROM gis_udp_bank, (SELECT @par := '51016') T
WHERE @par IS NOT NULL;

5. 向上递归

SELECT
    @par                                                                     AS par,
    (SELECT @par := GROUP_CONCAT(NAME) FROM demo_userm WHERE LOGONID = @par) AS son
FROM demo_userm, (SELECT @par := '10') T
WHERE @par IS NOT NULL;
par即为递归查询结果
查询结果:
| par | son |
| 10  | 9   |
| 9   | 1   |
| 1   | 0   |
| 0   | NULL|

所用函数介绍:

1、GROUP_CONCAT([distinct] colName)

一般用于取ID集合。
select  * from demo_userm t where NAME = 1;
| LOGONID | NAME |
| 2       | 1    |
| 8       | 1    |
| 9       | 1    |
select  GROUP_CONCAT(LOGONID) from demo_userm t where NAME = 1;
| GROUP\_CONCAT\(LOGONID\) |
| 2,8,9 |
-- 不去重取name集合
select  GROUP_CONCAT(NAME) from demo_userm t where NAME in (1,2,3,4,5,6,7);
| GROUP\_CONCAT\(NAME\) |
| 1,2,3,4,5,6,1,1 |
-- 去重再取name集合
select  GROUP_CONCAT(distinct NAME) from demo_userm t where NAME in (1,2,3,4,5,6,7);
| GROUP\_CONCAT\(distinct NAME\) |
| 1,2,3,4,5,6 |

2、FIND_IN_SET(colName, strs)

select * from demo_userm where  FIND_IN_SET(LOGONID, ('1,2,3,4'));
| LOGONID | NAME |
| 1       | 0    |
| 2       | 1    |
| 3       | 2    |
| 4       | 3    |
select * from demo_userm where  
FIND_IN_SET(LOGONID, (select  GROUP_CONCAT(LOGONID) from demo_userm t where NAME = 1));
| LOGONID | NAME |
| 2       | 1    |
| 8       | 1    |
| 9       | 1    |

3、#@是用户变量,@@是系统变量。

用户变量赋值有两种方式: 一种是直接用"=“号,另一种是用”:=“号。

其区别在于:

使用set命令对用户变量进行赋值时,两种方式都可以使用; # 用select语句时,只能用”:=“方式,因为select语句中,”="号被看作是比较操作符。

总结

到此这篇关于mysql通过@变量实现递归的文章就介绍到这了,更多相关mysql @变量实现递归内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 同时运行多个MySQL服务器的方法

    同时运行多个MySQL服务器的方法

    在同一台机器上运行多个有些情况下你可能想要在同一台机器上运行多个服务器。例如,你可能想要测试一个新的MySQL版本而让你现有生产系统的设置不受到干扰, 或你可能是想要为不同的客户提供独立的MySQL安装一个因特网服务供应商。
    2008-05-05
  • 区分MySQL中的空值(null)和空字符('''')

    区分MySQL中的空值(null)和空字符('''')

    这篇文章主要介绍了如何区分MySQL中的空值(null)和空字符(''),帮助大家更好的理解和使用MySQL数据库,感兴趣的朋友可以了解下
    2020-09-09
  • mysql 表空间及索引的查看方法

    mysql 表空间及索引的查看方法

    mysql 表空间及索引的查看方法,需要的朋友可以参考下。
    2011-07-07
  • 一文深入探讨MySQL是如何解决幻读问题

    一文深入探讨MySQL是如何解决幻读问题

    SQL标准中定义了4种隔离级别,分别是读未提交、读已提交、可重复读以及序列化。不同的隔离级别下,本文将重点探讨下MySQL是如何解决幻读问题的,需要的朋友可以跟着小编一起来探讨一下
    2023-07-07
  • MySQL中主键为0与主键自排约束的关系详解(细节)

    MySQL中主键为0与主键自排约束的关系详解(细节)

    这篇文章主要给大家介绍了关于MySQL中主键为0与主键自排约束的关系的相关资料,主要介绍的是其中的一些非常细的细节,对大家学习或者使用mysql具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-05-05
  • MySQL中or、in、union与索引优化详析

    MySQL中or、in、union与索引优化详析

    这篇文章主要给大家介绍了关于MySQL中or、in、union与索引优化的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用MySQL具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-12-12
  • mysql表分区的方式和实现代码示例

    mysql表分区的方式和实现代码示例

    通俗地讲表分区是将一个大表,根据条件分割成若干个小表,下面这篇文章主要给大家介绍了关于mysql表分区的方式和实现代码,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-02-02
  • MySQL外键设置的方法实例

    MySQL外键设置的方法实例

    这篇文章主要介绍了MySQL外键设置的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-04-04
  • MyEclipse连接MySQL数据库报错解决办法

    MyEclipse连接MySQL数据库报错解决办法

    我们现在一般网站都是利用的MySQL数据库搭建网站的,但是在网上看到很多网友吐槽数据库连接不上的问题,现在我就结合相关资料向提出一些我个人的见解,希望对大家解决问题有帮助
    2014-01-01
  • 浅谈Mysql insert on duplicate key 死锁问题定位与解决

    浅谈Mysql insert on duplicate key 死锁问

    本文介绍了在并发场景下的 insert on duplicate key update sql 出现的死锁,经过分析发现这种sql确实比较容易造成死锁,这篇文章就从分析死锁展开,到最终如何解决这样的问题 分享相应的思路,感兴趣的可以了解一下
    2022-05-05

最新评论