DBA优化SQL采用的WITH AS 用法简介
一、WITH AS简介
WITH AS的用法从oracle 9i新增的,官方文档也称之为:subquery factoring;在进行复杂的查询、统计等操作时使用with as 子句可以大大提高性能!
二、WITH AS用法
本人在oracle 11g 参考手册第1453页查询到官方对with as 的描述:
其中红色线框内给出了with as 的用法:oracle将其当作一个内联视图或者临时表来对待;
那么在进行复杂的查询时可以通过with as 对表数据进行处理,得到处理过后的结果作为一个临时表来操作,大大降低sql的复杂性;
官方文档给出with as 的用法如下:
由上图可知:其定义格式为:
WITH 名称1 [(别名1,别名2...)] AS ( 子查询 ) + 查询语句
以oracle数据库内置的用户scott中的dept,emp表来举例:
查询部门工资大于部门平均工资的部门信息
解决思路:
部门工资:按照部门进行分组,对每个部门下的员工工资进行求和,得到每个部门的工资;
部门平均工资:对上一步获得的部门工资进行就和后除以部门总数获得部门平均工资;
对以上的结果进行比较获得要求的部门信息
如果按照正常的思路去写,将会形成一个多重嵌套查询的sql,会是一个较为复杂的语句,如果利用with as 将部门工资和部门平均工资都
分别结算出来以后当作临时表,然后取得两个临时表中的工资数目进行比较很轻松获得要求的部门信息,其sql如下:
--统计部门工资大于平均工资的部门 WITH dept_cost AS (SELECT d.dname ,sum(nvl(p.sal,0)) AS dept_total FROM dept d,emp p WHERE p.deptno (+)= d.deptno GROUP BY d.dname), dept_avg AS (SELECT SUM(dept_total)/COUNT(1) AS sal_avg FROM dept_cost ) SELECT * FROM dept_cost a , dept_avg b WHERE a.dept_total > sal_avg ;
以上的dept_cost 就为部门工资,dept_avg为部门平均工资,通过with as 子句分别查询出来,然后通过一个select语句来操作两者获得结果如下:
DNAME DEPT_TOTAL SAL_AVG 1 ACCOUNTING 8750 7256.25 2 RESEARCH 10875 7256.25 3 SALES 9400 7256.25