oracle – 为什么这个查询有效?

前端之家收集整理的这篇文章主要介绍了oracle – 为什么这个查询有效?前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我有两个表,table_a(id,name)和table_b(id),让我们说在Oracle 12c上.

为什么此查询不返回异常?

select * from table_a where name in (select name from table_b);

据我所知,Oracle认为这是

select * from table_a where name = name;

但我没有得到的是为什么?

即使table_b没有名称列,查询在语法上也是正确的sql.原因是范围解决.

解析查询时,首先检查table_b是否具有名称列.因为它没有,所以检查table_a.仅当两个表都没有名称列时才会抛出错误.

最后,查询执行如下:

select a.* 
from table_a  a
where a.name in (select a.name 
                 from table_b  b
                );

至于查询将为table_a的每一行提供的结果,子查询(从table_b中选择名称) – 或(从table_b b中选择a.name) – 是一个具有相同a.name值的单个列的表,与table_b一样多的行.因此,如果table_b有1行或更多行,则查询运行如下:

select a.* 
from table_a  a
where a.name in (a.name,a.name,...,a.name) ;

要么:

select a.* 
from table_a  a
where a.name = a.name ;

要么:

select a.* 
from table_a  a
where a.name is not null ;

如果table_b为空,则查询将不返回任何行(thnx为@ughai以指示该可能性).

那个(你没有得到错误的事实)可能是所有列引用都应该以表名/别名为前缀的最好的原因.如果查询是:

select a.* from table_a where a.name in (select b.name from table_b);

你会马上得到错误.当省略表前缀时,不会发生这样的错误,尤其是在更复杂的查询中,更重要的是,不要注意.

另请参阅Oracle docs: Resolution of Names in Static SQL Statements中的内部捕获中的类似示例B-6以及“在SELECT和DML语句中避免内部捕获”中的建议段落:

Qualify each column reference in the statement with the appropriate table alias.

猜你在找的Oracle相关文章