常见的SQL等价改写

表结构

SQL> desc scott.emp
Name Type Nullable Default Comments
-------- ------------ -------- ------- --------
EMPNO NUMBER(4) Y
ENAME VARCHAR2(10) Y
JOB VARCHAR2(9) Y
MGR NUMBER(4) Y
HIREDATE DATE Y
SAL NUMBER(7,2) Y
COMM NUMBER(7,2) Y
DEPTNO NUMBER(2) Y

SQL> desc scott.dept
Name Type Nullable Default Comments
------ ------------ -------- ------- --------
DEPTNO NUMBER(2)
DNAME VARCHAR2(14) Y
LOC VARCHAR2(13) Y

半连接

select * from scott.emp a

where deptno in(select deptno from scott.dept b where  a.deptno = b.deptno)

等价

select * from scott.emp a

where exists(select 1 from scott.dept b where  a.deptno = b.deptno)

等价

select * from scott.emp a,(select deptno from scott.dept ) b

where a.deptno = b.deptno

 

反连接

select * from scott.emp a

where deptno not in (select deptno from scott.dept b where  a.deptno = b.deptno)

等价

select * from scott.emp a

where not exists(select 1 from scott.dept b where  a.deptno = b.deptno)

等价

select * from scott.emp a left join scott.dept b

on a.deptno = b.deptno

where b.deptno is null

 

 

标量子查询

select ename, (select dname from scott.dept b where a.deptno = b.deptno)
from scott.emp a
where empno = 7499

等价

select ename, dname
from scott.emp a left join scott.dept

on a.deptno = b.deptno

where empno = 7499

 

表反复查询

select ename from scott.emp

where empno in (select max(empno) from scott.emp group by deptno)

等价

select ename  from (

select ename ,

empno,

max(empno)over(partition by deptno) max_empno

 from scott.emp

) where empno=max_empno

 

 

select * from (
select count(*) from scott.emp where deptno=10),
(
select count(*) from scott.emp where deptno=20)

等价

select
sum(case when deptno=10 then 1 end),
sum(case when deptno=20 then 1 end)
from scott.emp

 

with可以改写sql中反复出现的查询体

 

 

在一些特定场景中(子查询加层次查询,rownum等),sql的某种写法,会导致执行计划有问题,走的是cost较高的执行计划。

所以需要sql改写,来让sql优化器能找出正确的执行计划。

 

还有就是sql的逻辑,多次查询同一个表与查询一次就能获得结果集,那个效率高很明显赛。

 

posted @ 2019-11-12 17:06  阿西吧li  阅读(841)  评论(0编辑  收藏  举报