一、Oracle的子查询分为两类分别是嵌套子查询和非嵌套子查询。所谓嵌套子查询是指,子查询是一个独立的查询不与外部查询相关,子查询将被先执行,而且只被执行一次,子查询执行完成后,再执行外部的查询,外部查询在执行过程中会使用到子查询的结果。下面我们来看嵌套子查询。
hr@OCM> set autot traceonly;
hr@OCM> SELECT employee_id,first_name,salary
2 FROM employees
3 WHERE
4 department_id=(SELECT department_id
5 FROM departments where department_name='IT');
nested query叫:inner query,main query叫outer query。
outer query是:SELECTemployee_id,first_name,salary
FROM employees;
Nested query是:SELECTdepartment_id
FROM departments where department_name='IT';
看上图的执行计划,是先执行:id=3(DEPARTMENTS做全表扫描,通过部门名称过滤出部门编号)即先执行nested query,然后再执行id=2(对EMPLOYEES表的DEPARTMENT_ID列上的索引做范围扫描,然后能索引的键值找到EMPLOYEES表中的编号、姓名、薪水),取后执行outer query。最后SELECT出来的数据来自于outer query。
通过上面的分析:子查询可以向主查询提供常量查询条件,子查询被优先执行,产生的结果作为主查询的过滤条件,然后主果询以这个条件过滤出结果,返回给用户数据。
二、上面的语句也可以转化成表连接的方式:
hr@OCM> SELECT employee_id,first_name,salary
2 FROM employees e,departments d
3 WHERE e.department_id=d.department_id
4 AND department_name='IT';
三、也可以用子查询非嵌套化,解除构成子查询的查询语句与主查询语句的嵌套关系或者通过表的连接方式替代子查询,以获得良的执行速度。事实上,几乎大部分的子查询都要经过转化。在子查询非嵌套化不可能实现时,优化器会制定把子查询放在最优先或最后位置 执行的独立执行计划。此时,这个查询语句的执行速度会随着子查询被放在最优先或最后位置执行的顺序不同而不同。
hr@OCM> SELECT department_id, last_name, salary
2 FROM employees x
3 WHERE salary > (SELECT AVG(salary)
4 FROM employees
5 WHERE x.department_id = department_id)
6 ORDER BY department_id;
总结:如果优化器无法将其连接转为表连接查询,而只能为其中的主查询和子查询制定相应的执行计划 。如果处理类型为需要将子查询的结果提供给主查询的列,且在子查询中能够使用索引,那么优化器就将制定优先执行子查询的执行计划。如果处理类型为在主查询被执行的同时验证子查询的FILTER,或在读取了各个集合数据之后再对其进行合并,那么就会按照hash连接的方式来执行。
A答案是说:先执行outer query,再执行nested query
B答案是说:先执行nested query,再执行outer query
C答案是说:只有返回了nested query的结果,才能执行outer query
D答案是说:nested query返回的结果能估算出outer query中返回的每一行。
库题的答案是:AD,个人觉得BD答案更合适
为什么B错呢?这个答案有点疑异,我们大家一起来讨论!!!!