关于Oracle在where子句中引用列别名问题的分析
2015年08月17日 16:02:45 u010752082 阅读数:3857
SQL> SELECT SAL SALARY FROM EMP WHERE SALARY<2500;
Warning: connection was lost and re-established
SELECT SAL SALARY FROM EMP WHERE SALARY<2500
ORA-00904: "SALARY": 标识符无效
在ORACLE中,在WHERE子句中引用列别名会出错,这是为什么呢?
请看SELECT 语句的执行顺序:
1. from语句
2. where语句(结合条件)
3. start with语句
4. connect by语句
5. where语句
6. group by语句
7. having语句
8. model语句
9. select语句
10. union、minus、intersect等集合演算演算
11. order by语句
SELECT 语句在WHERE语句后面才执行,而列的别名是在SELECT 时才生成的,故在WHERE子句中根本看不到列的别名,自然无法引用列的别名了。
如果想用列的别名就要这样:
SELECT SALARY FROM (SELECT SAL SALARY FROM EMP) WHERE SALARY<2500;
---------------------------------------------------------------------------------------------------------------------------------------
问题
前面已经使用了别名为查询提供更有意义的列名,而且也使用WHERE子句将一些数据排除掉,然而,我们还想在WHERE子句中引用别名。
select sal as salary, comm as commission
from emp
where salary < 5000
解决方案
将查询作为内联视图就可以就可以引用其中别名的列了:
select *
from (
select sal as salary, comm as commission
from emp ) x
where salary < 5000
讨论
在这个简单的示例中,可以不使用内联视图,也不用在WHERE子句中直接引用COMM或SAL而得到相同的结果。本方案介绍的方法在下列情形的WHERE子句都可以使用:
聚集函数
标量子查询
视窗函数
别名
将取别名的查询作为内联视图,便可以在外部查询中引用其别名列,为什么要这么做呢?WHERE子句是在SELECT子句之前进行处理的,在处理求解查询“问题”WHERE子句之前,SALARY和COMMISSION并不存在,要到WHERE子句处理完成之后,别名才有效。然而,FROM子句是在WHERE子句之前进行处理的。将原始查询放在FROM之句之中,那么,在最外层的WHERE子句之前,以及最外层的WHERE子句“看到”别名之前,就已经生成了查询结果。如果表中的列没有特别命名的话这是一个非常有用的技巧。
注意 这个解决方案中的内联视图别名为X。并非所有数据库都需要内联视图显示给内联视图取别名,但有一些数据库是这样的。所有的数据库都接受这种方式。