Skip to content

Latest commit

 

History

History
92 lines (70 loc) · 2.09 KB

sql-statement-with.md

File metadata and controls

92 lines (70 loc) · 2.09 KB
title summary
WITH
TiDB 数据库中 WITH (公共表表达式) 的使用概况。

WITH

公共表表达式 (CTE) 是一个临时的中间结果集,能够在 SQL 语句中引用多次,提高 SQL 语句的可读性与执行效率。在 TiDB 中可以通过 WITH 语句使用公共表表达式。

语法图

WithClause:

WithClause ::=
        "WITH" WithList
|       "WITH" "RECURSIVE" WithList

WithList:

WithList ::=
        WithList ',' CommonTableExpr
|       CommonTableExpr

CommonTableExpr:

CommonTableExpr ::=
        Identifier IdentListWithParenOpt "AS" SubSelect

IdentListWithParenOpt:

IdentListWithParenOpt ::=
( '(' IdentList ')' )?

示例

非递归的 CTE:

WITH cte AS (SELECT 1, 2) SELECT * FROM cte t1, cte t2;
+---+---+---+---+
| 1 | 2 | 1 | 2 |
+---+---+---+---+
| 1 | 2 | 1 | 2 |
+---+---+---+---+
1 row in set (0.00 sec)

递归的 CTE:

WITH RECURSIVE cte(a) AS (SELECT 1 UNION SELECT a+1 FROM cte WHERE a < 5) SELECT * FROM cte;
+---+
| a |
+---+
| 1 |
| 2 |
| 3 |
| 4 |
| 5 |
+---+
5 rows in set (0.00 sec)

MySQL 兼容性

  • 在严格模式下,当递归部分算出的数据长度超过初始部分的数据长度时,TiDB 会返回警告,而 MySQL 会返回错误。在非严格模式下,TiDB 与 MySQL 行为一致。
  • 递归 CTE 所使用的数据类型由初始部分决定。初始部分的数据类型在某些情况(例如函数)下与 MySQL 并不完全一致。
  • 多个 UNION / UNION ALL 情况下,MySQL 不允许 UNION 后面加 UNION ALL,TiDB 允许。
  • 如果 CTE 的定义存在问题,TiDB 会报错,而 MySQL 在未引用的情况下不报错。

另请参阅