對於邏輯複雜的sql,with能夠大大減小臨時表的數量,提高代碼的可讀性、可維護性
MySQL 8.0終於開始支持with語句了,對於複雜查詢,能夠不用寫那麼多的臨時表了。
能夠查看官方文檔【點擊跳轉】mysql
官方第一個示例,能夠看出該查詢語句建立了cte1
,cte2
,cte3
,cte4
這4個臨時表,後面的臨時表依賴前面的臨時表數據。
最後一行爲最終查詢結果,實際ct4
由於ct3
結果包含3行數據,可是使用MAX
,MIN
獲得一行結果。sql
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
,查詢臨時表,所以查詢出的結果就顯而易見了express
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個不同的查詢,查詢結果都同樣。
值得注意的是臨時表裏面的多個查詢列數量和類型必須同樣,否則會報錯。sqlserver
這個是將臨時表列名指定在第一行 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;
根據官方文檔,臨時表的語法模板以下,是能夠有不少行的查詢共同組成。3d
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 ]
官方文檔還列出了,使用臨時表時能夠增刪改查新表,具體能夠去閱讀官方文檔。code
關於遞歸的練習主要用於表裏麪包含父節點id之類的,詳情能夠參考下面的練習。
定義下面這樣的表,存儲每一個區域(省、市、區)的id,名字及上級區域的pidserver
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;
執行結果:blog
分析結果包含第一行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
查詢是爲了不出現嵌套的子查詢,每一個查詢結果均可以是一個臨時表,而後總查詢能夠用到全部臨時表的數據。
而後就是遞歸查詢,能夠解決樹形接口的狀況,數據有父子層級的那種。接口