国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 數(shù)據(jù)庫 > MySQL > 正文

教你如何使用MySQL8遞歸的方法

2024-07-24 12:41:40
字體:
供稿:網(wǎng)友

之前寫過一篇 MySQL通過自定義函數(shù)的方式,遞歸查詢樹結(jié)構(gòu),從MySQL 8.0 開始終于支持了遞歸查詢的語法

CTE

首先了解一下什么是 CTE,全名 Common Table Expressions

WITH cte1 AS (SELECT a, b FROM table1), cte2 AS (SELECT c, d FROM table2)SELECT b, d FROM cte1 JOIN cte2WHERE cte1.a = cte2.c;

cte1, cte2 為我們定義的CTE,可以在當(dāng)前查詢中引用

可以看出 CTE 就是一個臨時結(jié)果集,和派生表類似,二者的區(qū)別這里不細(xì)說,可以參考下MySQL開發(fā)文檔:https://dev.mysql.com/doc/refman/8.0/en/with.html#common-table-expressions-recursive-examples

遞歸查詢

先來看下遞歸查詢的語法

WITH RECURSIVE cte_name AS(  SELECT ...   -- return initial row set  UNION ALL / UNION DISTINCT  SELECT ...   -- return additional row sets)SELECT * FROM cte;
定義一個CTE,這個CTE 最終的結(jié)果集就是我們想要的 ”遞歸得到的樹結(jié)構(gòu)",RECURSIVE 代表當(dāng)前 CTE 是遞歸的 第一個SELECT 為 “初始結(jié)果集” 第二個SELECT 為遞歸部分,利用 "初始結(jié)果集/上一次遞歸返回的結(jié)果集" 進(jìn)行查詢得到 “新的結(jié)果集” 直到遞歸部分結(jié)果集返回為null,查詢結(jié)束 最終UNION ALL 會將上述步驟中的所有結(jié)果集合并(UNION DISTINCT 會進(jìn)行去重),再通過 SELECT * FROM cte; 拿到所有的結(jié)果集

遞歸部分不能包括:

聚合函數(shù)例如 SUM() GROUP BY ORDER BY LIMIT DISTINCT

上面的講解可能有點(diǎn)抽象,通過例子慢慢來理解

WITH RECURSIVE cte (n) AS -- 這里定義的n相當(dāng)于結(jié)果集的列名,也可在下面查詢中定義( SELECT 1 UNION ALL SELECT n + 1 FROM cte WHERE n < 5)SELECT * FROM cte;-- result+------+| n  |+------+|  1 ||  2 ||  3 ||  4 ||  5 |+------+
初始結(jié)果集為 n =1 這時候看遞歸部分,第一次執(zhí)行 CTE結(jié)果集即是 n =1,條件發(fā)現(xiàn)并不滿足 n < 5,返回 n + 1 第二次執(zhí)行遞歸部分,CTE結(jié)果集為 n = 2,遞歸... 直至條件不滿足 最后合并結(jié)果集

EXAMPLE

最后來看一個樹結(jié)構(gòu)的例子

CREATE TABLE `c_tree` ( `id` int(11) NOT NULL AUTO_INCREMENT, `cname` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT NULL, `parent_id` int(11) DEFAULT NULL, PRIMARY KEY (`id`)) ENGINE=InnoDB AUTO_INCREMENT=13 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
mysql> select * from c_tree;+----+---------+-----------+| id | cname  | parent_id |+----+---------+-----------+| 1 | 1    |     0 || 2 | 2    |     0 || 3 | 3    |     0 || 4 | 1-1   |     1 || 5 | 1-2   |     1 || 6 | 2-1   |     2 || 7 | 2-2   |     2 || 8 | 3-1   |     3 || 9 | 3-1-1  |     8 || 10 | 3-1-2  |     8 || 11 | 3-1-1-1 |     9 || 12 | 3-2   |     3 |+----+---------+-----------+
mysql> WITH RECURSIVE tree_cte as(  select * from c_tree where parent_id = 3  UNION ALL  select t.* from c_tree t inner join tree_cte tcte on t.parent_id = tcte.id)SELECT * FROM tree_cte;+----+---------+-----------+| id | cname  | parent_id |+----+---------+-----------+| 8 | 3-1   |     3 || 12 | 3-2   |     3 || 9 | 3-1-1  |     8 || 10 | 3-1-2  |     8 || 11 | 3-1-1-1 |     9 |+----+---------+-----------+
發(fā)表評論 共有條評論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 南和县| 桐城市| 汉源县| 汪清县| 贡觉县| 建阳市| 隆昌县| 临沂市| 昌吉市| 西丰县| 二连浩特市| 普兰县| 芮城县| 泰宁县| 富锦市| 航空| 武胜县| 宜章县| 定兴县| 西峡县| 板桥市| 县级市| 潮安县| 安西县| 晋江市| 东明县| 济源市| 湛江市| 井研县| 梧州市| 沐川县| 建阳市| 侯马市| 龙南县| 利津县| 滦平县| 新建县| 鹤壁市| 改则县| 贵州省| 田东县|