Oracle!你必须要知道的Knowledge points
子查询
什么是子查询
当查询中的限制条件需要另一个查询提供时,我们可以把两个查询语句嵌套起来,提供条件的查询语句作为子查询。
子查询,也叫内部查询,先于主查询执行,子查询的结果被用于主查询。
子查询分为单行子查询、多行子查询和多列子查询,单行子查询作为判断条件时用单行运算符,多行子查询和多列子查询作为判断条件时用多行运算符。
单行运算符:>、=、>=、<、<>、<=
多行运算符:in、all、any
子查询可以嵌套在哪里
Where子句:作为非分组函数筛选判断的条件
From子句:将子查询的查询结果作为一张表来使用
Having子句:作为分组函数筛选判断的条件
子查询使用规范
子查询需要用括号括起来
当子查询放在from子句中时,是将查询到的结果作为一张表来使用,需给别名
单行子查询用单行运算符,多行子查询用多行运算符
子查询放在运算符右边
单行子查询
单行子查询返回一行一列,使用单行运算符进行连接
--显示和雇员7369从事相同工作并且工资大于雇员7876的雇员的姓名和工作。select ename,jobfrom emp where job = (select job from emp where empno =7369)and sal > (select sal from emp where empno = 7876);--查询工资最低的员工姓名,岗位及工资(子查询中使用了分组函数)select ename,job,salfrom empwhere sal = (select min(sal) from emp)--查询部门最低工资比20部门最低工资高的部门编号及最低工资(子查询嵌套在having子句)select deptno,min(sal)from empgroup by deptnohaving min(sal) > (select min(sal) from emp where deptno = 20 );
多行子查询
多行子查询返回一行或多行记录,使用多行运算符进行连接
--查询是经理的员工姓名、工资select ename,salfrom empwhere empno in (select mgr from emp)
ANY的使用
表示和子查询的任意一行结果进行比较,有一个满足条件即可。
any:大于子查询结果的任意一个,即大于最小值即可。
=any:等于子查询结果的任意一个,即等一任意一个即可,相当于in。
--查询部门编号不为10,且工资比10部门任意一名员工工资高的员工编号,姓名,职位,工资。select empno,ename,job,salfrom empwhere deptno <> 10 and sal > any(select sal from emp where deptno = 10);
ALL的使用
表示和子查询的所有行结果进行比较,每一行必须都满足条件。
all:表示大于子查询结果集中的所有行,即大于最大值。= all :表示等于子查询结果集中的所有行,即等于所有值。
--查询部门编号不为10,且工资比10部门所有员工工资低的员工编号,姓名,职位,工资。select empno,ename,job,salfrom empwhere deptno <> 10 and sal > all(select sal from emp where deptno = 10);
多列子查询
在一个表达式内同时和子查询的多个列进行比较,即子查询返回的记录含多列
--查询和1981年入职的任意一个员工的部门和职位完全相同员工姓名、部门、职位、入职日期,不包括1981年入职员工。select ename,deptno,job,hiredatefrom empwhere (deptno,job) in (select deptno,job from emp where to_char(hiredate,'YYYY') = '1981')and to_char(hiredate,'YYYY') <> '1981';
--查询和1981年入职的任意一个员工的部门或职位相同员工姓名、部门、职位、入职日期,不包括1981年入职员工select ename,deptno,job,hiredatefrom empwhere job in (select job from emp where to_char(hiredate,'YYYY') = '1981')or deptno in (select deptno from emp where to_char(hiredate,'YYYY') = '1981')and to_char(hiredate,'YYYY') <> '1981';
子查询中的空值
子查询结果中含有空值对主查询会有怎样的影响?
--查询不是经理的员工姓名select enamefrom empwhere empno not in (select mgr from emp);
子查询的结果中有一条空值,这条空值导致主查询没有记录返回,这是因为所有的条件和空值比较结果都是空值,因此无论什么时候只要空值
有可能成为子查询结果集合中的一部分,就不能使用 not in 运算符。
如果子查询的结果中包含空值,又想使用 not in 运算符该怎么办?
--查询不是经理的员工姓名select enamefrom empwhere empno not in (select nvl(mgr,0) from emp); --使用nvl()函数去除空值影响
在from子句中使用子查询
在from子句中使用子查询,相当于把子查询的查询结果作为一张表来使用,在使用时建议加上相应的别名
--查询比自己部门平均工资高的员工姓名,工资,部门编号,部门平均工资select e.ename,e.sal,e.deptno,w.avgsalfrom emp e,(select avg(sal) as avgsal deptno from emp group by deptno) wwhere e.deptno = w.deptno and e.sal > w.avgsal;
ROWNUM
rownum是一个伪列,顾名思义,就是一个类似于表中的列,而实际并没有存储在表中的特殊列;
rownum的功能是在每次查询时,返回结果集的顺序号,这个顺序号是在记录输出时才一步一步产生的,第一行显示为1,第二行为2,以此类推。
select rownum.emp.* from emp;
关于rownum需要知道的几点
- rownum是在记录输出时才生成,且总是从1开始
- rownum只能执行<、<=运算,不能执行>、>= 或区间运算between ... and ...
- rownum和order by一起使用时,因为rownum在记录输出时生成,而order by子句在最后执行,所以当两者一起使用时,需要注意rownum实际是已经被排了序的rownum。
TOP-N查询
实现表中按照某个列排序,输出最大或最小的N条记录功能
Top-N语法
ASC:升序排序,查询最小的N条记录
DESC:降序排序,查询最大的N条记录
--查询入职日期最早的前5名员工姓名,入职日期。select rownum,ename,hiredatefrom (select ename,hiredate from emp order by hiredate)where rownum <= 5;
--查询工作在CHICAGO并且入职日期最早的前2名员工姓名,入职日期select rownum,ename,hiredatefrom (select ename,hiredate from emp where deptno = (select deptno from dept where loc = 'CHICAGO') order by hiredate)where rownum <= 2;
分页查询
利用ROWNUM的特性,可以实现数据库端的分页查询
语法格式
--按照每页显示5条记录,分别查询第1页,第2页,第3页信息,要求显示员工姓名、入职日期、部门名称。select b.*from (select rownum rn,ename,hiredate,dname from emp,dept where emp.deptno = dept.deptno and rownum <= 1*5) bwhere rn > (1-1)*5;
select b.*from (select rownum rn,ename,hiredate,dname from emp,dept where emp.deptno = dept.deptno and rownum <= 2*5) bwhere rn > (2-1)*5;
select b.*from (select rownum rn,ename,hiredate,dname from emp,dept where emp.deptno = dept.deptno and rownum <= 3*5) bwhere rn > (3-1)*5;
排序后分页
排序->rownum->select输出最后结果
--按照每页显示5条记录,分别查询工资最高的第1页,第2页,第3页信息,要求显示员工姓名、入职日期、部门名称、工资。select *from (select rownum rn,b.* from (select ename,hiredate,dname,sal from emp,dept where emp.deptno = dept.deptno order by sal desc) b where rownum <= 1*5 )where rn > (1-1) * 5;
select *from (select rownum rn,b.* from (select ename,hiredate,dname,sal from emp,dept where emp.deptno = dept.deptno order by sal desc) b where rownum <= 2*5 )where rn > (2-1) * 5;
select *from (select rownum rn,b.* from (select ename,hiredate,dname,sal from emp,dept where emp.deptno = dept.deptno order by sal desc) b where rownum <= 3*5 )where rn > (3-1) * 5;