目錄
- 前言
- 示例
- 練習
- 總結
前言
對于邏輯復雜的sql,with可以大大減少臨時表的數量,提升代碼的可讀性、可維護性
MySQL 8.0終于開始支持with陳述句了,對于復雜查詢,可以不用寫那么多的臨時表了,
可以查看官方檔案【點擊跳轉】
示例
官方第一個示例,可以看出該查詢陳述句創建了cte1,cte2,cte3,cte4這4個臨時表,后面的臨時表依賴前面的臨時表資料,
最后一行為最終查詢結果,實際ct4因為ct3結果包含3行資料,但是使用MAX,MIN得到一行結果,
WITH cte1(txt) AS (SELECT "This "),
cte2(txt) AS (SELECT CONCAT(cte1.txt,"is a ") FROM cte1),
cte3(txt) AS (SELECT "nice query" UNION
SELECT "query that rocks" UNION
SELECT "query"),
cte4(txt) AS (SELECT concat(cte2.txt, cte3.txt) FROM cte2, cte3)
SELECT MAX(txt), MIN(txt) FROM cte4;
+----------------------------+----------------------+
| MAX(txt) | MIN(txt) |
+----------------------------+----------------------+
| This is a query that rocks | This is a nice query |
+----------------------------+----------------------+
1 row in set (0,00 sec)
官方第二個示例是遞回的用法,根據閱讀檔案,我分析下面查詢結果如下,
首先定義一個臨時表my_cte
分析SELECT 1 AS n,這個是決定臨時表的列名為n,值為1
然后SELECT 1+n FROM my_cte WHERE n<10,這個是遞回查詢n<10,并將1+n作為結果填充臨時表
最終使用SELECT * FROM my_cte,查詢臨時表,因此查詢出的結果就顯而易見了
WITH RECURSIVE my_cte AS
(
SELECT 1 AS n
UNION ALL
SELECT 1+n FROM my_cte WHERE n<10
)
SELECT * FROM my_cte;
+------+
| n |
+------+
| 1 |
| 2 |
| 3 |
| 4 |
| 5 |
| 6 |
| 7 |
| 8 |
| 9 |
| 10 |
+------+
10 rows in set (0,00 sec)
根據我的理解寫了如下2個不一樣的查詢,查詢結果都一樣,
值得注意的是臨時表里面的多個查詢列數量和型別必須一樣,不然會報錯,
這個是將臨時表列名指定在第一行
WITH RECURSIVE my_cte(a,b,c) AS
(
SELECT 1,1,1
UNION ALL
SELECT 1+a,2+b,3+c FROM my_cte WHERE a<10
)
SELECT * FROM my_cte;
這個第一行沒有指定列名,然后列名由第一個查詢回傳結果確定
WITH RECURSIVE my_cte AS
(
SELECT 1 AS a,1 AS b,1 AS c
UNION ALL
SELECT 1+a,2+b,3+c FROM my_cte WHERE a<10
)
SELECT * FROM my_cte;
根據官方檔案,臨時表的語法模板如下,是可以有很多行的查詢共同組成,
WITH RECURSIVE cte_name [list of column names ] AS
(
SELECT ... <-- specifies initial set
UNION ALL
SELECT ... <-- specifies initial set
UNION ALL
...
SELECT ... <-- specifies how to derive new rows
UNION ALL
SELECT ... <-- specifies how to derive new rows
...
)
[, any number of other CTE definitions ]
官方檔案還列出了,使用臨時表時可以增刪改查新表,具體可以去閱讀官方檔案,
練習
關于遞回的練習主要用于表里面包含父節點id之類的,詳情可以參考下面的練習,
定義下面這樣的表,存盤每個區域(省、市、區)的id,名字及上級區域的pid
CREATE TABLE tb(id VARCHAR(3), pid VARCHAR(3), name VARCHAR(64));
INSERT INTO tb VALUES('002', 0, '浙江省');
INSERT INTO tb VALUES('001', 0, '廣東省');
INSERT INTO tb VALUES('003', '002', '衢州市');
INSERT INTO tb VALUES('004', '002', '杭州市');
INSERT INTO tb VALUES('005', '002', '湖州市');
INSERT INTO tb VALUES('006', '002', '嘉興市');
INSERT INTO tb VALUES('007', '002', '寧波市');
INSERT INTO tb VALUES('008', '002', '紹興市');
INSERT INTO tb VALUES('009', '002', '臺州市');
INSERT INTO tb VALUES('010', '002', '溫州市');
INSERT INTO tb VALUES('011', '002', '麗水市');
INSERT INTO tb VALUES('012', '002', '金華市');
INSERT INTO tb VALUES('013', '002', '舟山市');
INSERT INTO tb VALUES('014', '004', '上城區');
INSERT INTO tb VALUES('015', '004', '下城區');
INSERT INTO tb VALUES('016', '004', '拱墅區');
INSERT INTO tb VALUES('017', '004', '余杭區');
INSERT INTO tb VALUES('018', '011', '金東區');
INSERT INTO tb VALUES('019', '001', '廣州市');
INSERT INTO tb VALUES('020', '001', '深圳市');
WITH RECURSIVE cte AS (
SELECT id,name FROM tb WHERE id='002'
UNION ALL
SELECT k.id, CONCAT(c.name,'->',k.name) AS name FROM tb k INNER JOIN cte c ON c.id = k.pid
) SELECT * FROM cte;
執行結果:

分析結果包含第一行SELECT id,name FROM tb WHERE id='002'的資料,此時表中只有一行資料
然后連表查詢SELECT k.id, CONCAT(c.name,'->',k.name) AS name FROM tb k INNER JOIN cte c ON c.id = k.pid,遞回的將父節點資料放入臨時表
最終查詢出來的就是遞回的結果,
總結
通過閱讀官方檔案,我知道了
WITH查詢是為了避免出現嵌套的子查詢,每個查詢結果都可以是一個臨時表,然后總查詢可以用到所有臨時表的資料,
然后就是遞回查詢,可以解決樹形介面的情況,資料有父子層級的那種,
轉載請註明出處,本文鏈接:https://www.uj5u.com/shujuku/292295.html
標籤:其他
