概述
WITH AS短语,也叫做子查询部分(subquery factoring),是用来定义一个SQL片断,该SQL片断会被整个SQL语句所用到。这个语句,即为公用表表达式(CTE,common table expression),比如with A as (select * from class) select * from A
,先执行select * from class
得到一个结果,记录为A,再执行select * from A
语句。A 表只是一个别名。对于大批量的SQL数据,起优化作用。
查询
更新
总结
- 使用with子句可以让子查询重用相同的with查询块,通过select调用(with子句只能被select查询块引用),一般在with查询用到多次情况下。在引用的select语句之前定义,同级只能定义with关键字只能使用一次,逗号分割。
- with子句的返回结果存到用户的临时表空间中,只做一次查询,反复使用,提高效率。
- 在同级select前有多个查询定义的时候,第1个用with,后面的不用with,逗号隔开。
- 最后一个with子句与下面的查询之间不能有逗号,只通过右括号分割,with 子句的查询必须用括号括起来
- 如果定义with子句,而在查询中不使用,会报
ora-32035
错误:未引用在with子句中定义的查询名。(至少一个with查询的name未被引用,解决方法是移除未被引用的with查询),只要后面有引用的就可以,不一定非要在主查询中引用,后面的with查询引用也可以 - 前面的with子句定义的查询在后面的with子句中可以使用。但是一个with子句内部不能嵌套with子句。
- 当一个查询块名字和一个表名或其他的对象相同时,解析器从内向外搜索,优先使用子查询块名字。
- with查询的结果列有别名,引用时必须使用别名或
*
。
扩展
不同的SQL dialect,对with as的支持不一样。
Hive
clickhouse
clickhouse-with-as CK提供WITH关键字来支持CTE,WITH结果子句可以在其余部分中使用SELECT查询。
CK with as
的限制:
- 不支持递归查询
- 当在section中使用子查询时,它的结果应该是只有一行的标量
- Expression的结果在子查询中不可用
例
- 使用常量表达式作为 variable
- 从SELECT子句列表中逐出sum(bytes)表达式结果
- 使用标量子查询的结果
- 在子查询中重用表达式
作为子查询中表达式使用的当前限制的解决方法,
PostgreSQL
PostgreSQL对with as
的支持也很完备,如支持多个with子句。实例:
使用with查询出固定时间内充值的玩家id和期间充值总数,使用with查询出指定时间之前充值的玩家的id和历史充值总数,再查询出u1中所有id对应的u2中的数据,以及pinfo中的sid。
可以在WITH中使用数据修改语句(INSERT、UPDATE或DELETE),允许你在同一个查询中执行多个操作:
参考
PostgreSQL 9.5.3 with as querySQL with as 用法