WITH

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

语法图

WithClause:

  1. WithClause ::=
  2. "WITH" WithList
  3. | "WITH" recursive WithList

WithList:

  1. WithList ::=
  2. WithList ',' CommonTableExpr
  3. | CommonTableExpr

CommonTableExpr:

  1. CommonTableExpr ::=
  2. Identifier IdentListWithParenOpt "AS" SubSelect

IdentListWithParenOpt:

  1. IdentListWithParenOpt ::=
  2. ( '(' IdentList ')' )?

示例

非递归的 CTE:

  1. WITH CTE AS (SELECT 1, 2) SELECT * FROM cte t1, cte t2;
  1. +---+---+---+---+
  2. | 1 | 2 | 1 | 2 |
  3. +---+---+---+---+
  4. | 1 | 2 | 1 | 2 |
  5. +---+---+---+---+
  6. 1 row in set (0.00 sec)

递归的 CTE:

  1. WITH RECURSIVE cte(a) AS (SELECT 1 UNION SELECT a+1 FROM cte WHERE a < 5) SELECT * FROM cte;
  1. +---+
  2. | a |
  3. +---+
  4. | 1 |
  5. | 2 |
  6. | 3 |
  7. | 4 |
  8. | 5 |
  9. +---+
  10. 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 在未引用的情况下不报错。

另请参阅