多表查询

笛卡尔积

同时查询多张表时,每张表的每条数据都要和其它表的每条数据做组合。如下栗子,我们发现产生的总记录数是 56 条,还发现 emp 表是 14 条,dept 表是 4 条,56 条正是 emp 表和 dept 表的记录数的乘积,这就是笛卡尔积。

select count(1) from dept;

select count(1) from emp;

select count(1) from emp,dept;

例:

连接条件类型

  • 等值连接
  • 不等值连接
  • 外链接
  • 自连接

多表基本查询

使用一张以上的表做查询就是多表查询,而多表查询一般则需要通过多表连接来实现。

如果多张表一起进行查询而且每张表的数据很大的话那么该查询的笛卡尔积就会变得非常大,对性能造成影响。而我们仅需要笛卡尔积中部分对我们有用的数据,这时我们可以使用关联查询。

在 emp(员工) 表和 dept(部门) 表中我们会发现有一个共同的字段是 depno,depno 就是两张表关联的字段,用来描述一个员工属于哪个部门。我们可以使用这个字段来做限制条件。

两张表的关联查询字段一般一个是其中一张表的主键,另一个是另一张表的外键。

select e1.ename 员工名字,d1.dname 部门名称,d1.loc 部门地址 from dept d1,emp e1 where d1.deptno = e1.deptno;

例:查询出员工名字和其所属部门的名称、地址

select e1.ename 员工名字,e2.ename 领导名字 from emp e1,emp e2 where e1.mgr=e2.empno;

例:查询出每个员工名字和该员工的领导名字

select
e1.ename 员工名字,
d1.dname 所属部门名称,
decode(s1.grade,1,'一级',2,'二级',3,'三级',4,'四级',5,'五级') 员工工资等级,
e2.ename 领导名字,
decode(s2.grade,1,'一级',2,'二级',3,'三级',4,'四级',5,'五级') 领导工资等级
from
emp e1,dept d1,salgrade s1,emp e2,salgrade s2
where
e1.deptno=d1.deptno
and
e1.sal between s1.losal and s1.hisal
and
e1.mgr=e2.empno
and
e2.sal between s2.losal and s2.hisal;

例:查询出每个员工的名字、部门名称、工资等级和该员工的领导名字、工资等级

外连接

在上面‘查询出员工名字和其所属部门的名称、地址’示例中,其中有一个部门是没有员工的,可还是要求要把这个部门显示出来,这时就可以使用右连接:

-- 使用 +
select e1.ename 员工名字,d1.dname 部门名称 from emp e1,dept d1 where e1.deptno(+)=d1.deptno;
-- 使用 right join
select e1.ename 员工名字,d1.dname 部门名称 from emp e1 right join dept d1 on e1.deptno=d1.deptno;

右连接

在上面‘查询出每个员工名字和该员工的领导名字’示例中,其中有一个员工是没有领导的,可还是要求将这个员工显示出来,这时就可以使用左连接:

-- 使用 +
select e1.ename 员工名字,e2.ename 领导名字 from emp e1,emp e2 where e1.mgr=e2.empno(+);
-- 使用 left join
select e1.ename 员工名字,e2.ename 领导名字 from emp e1 left join emp e2 on e1.mgr=e2.empno;

右连接

注意:'+' 方式外连接是 Oracle 数据库独有的。

子查询

select
e1.empno 员工编号,
e1.ename 员工名字,
e1.job 工作,
e1.sal 工资
from
emp e1
where
e1.sal>(select sal from emp e2 where empno=7654)
and
e1.job=(select job from emp e3 where empno=7788)

例:查询出比员工 7654 的工资高,同时从事和 7788 相同工作的员工编号、名字、工作、工资

select
t_min.min_sal 最低工资,
e1.ename 员工名称,
d1.dname 部门名称
from
dept d1,
(select deptno,min(sal) min_sal from emp group by deptno) t_min,
emp e1
where d1.deptno = t_min.deptno
and e1.sal = t_min.min_sal

例:查询每个部门的最低工资、最低工资的员工名称和该员工所属的部门名称

-- 查询有员工的部门
select * from dept d1 where exists (select * from emp e1 where d1.deptno=e1.deptno);

例(exists):查询出有员工的部门

-- 查询员工 7788、7844 的领导信息
select * from emp where empno in(select mgr from emp where empno in (7788,7844));

例(in):查询员工 7788、7844 的领导信息

-- 查询比部门编号为 20 的部门中的所有员工工资都高的员工
select * from emp e1 where e1.sal>all(select sal from emp e2 where e2.deptno=20);

例(all):查询比部门编号为 20 的部门中的所有员工工资都高的员工

-- 查询员工编号为 7844、7900 的员工
select * from emp e1 where e1.empno=any(7844,7900);

例(any):查询员工编号为 7844、7900 的员工

null 值问题:

-- 查询是领导的员工
select * from emp where empno in (select mgr from emp);

-- 查询不是领导的员工
select * from emp where empno not in (select mgr from emp);

--此时会发现结果为空,查看会发现子查询中存在了 null 值:
select mgr from emp;

--在 Oracle 中,任何条件与 null 比较结果都为 null ,所以使用在对一个子查询使用 not in 时,应先过滤子查询结果中的 null 值:
select * from emp where empno not in (select mgr from emp where mgr is not null);

例(not in):查询不是领导的员工

集合运算

-- union :去重排序再合并
(select * from emp where sal>1500)
union
(select * from emp where deptno=20)

-- union all :不去重直接合并所有
(select * from emp where sal>1500)
union all
(select * from emp where deptno=20)

例(union&union all):查询工资大于 1500 或者是编号为 20 的部门下的员工

--查询工资大于1500,且在20号部门下的员工
-- intersect :交集运算
(select * from emp where sal>1500)
intersect
(select * from emp where deptno=20);

例(intersect):查询工资大于1500,且在20号部门下的员工

-- minus :差集运算
(select * from emp where to_char(hiredate,'yyyy')='')
minus
(select * from emp where job='PRESIDENT' or job='MANAGER')

例(minus):查询 1981 年的入职员工,不包括总裁和经理

集合运算的注意事项:

  • 列类型必须一致。
  • 列的数量必须一致,如果不足,可用空值填充。

练习

--rownum 是一个伪列,由系统自动生成,用来表示行号。rownum 是 Oracle 中特有的用来表示行号的列,起始值为 1,在查询出结果后再加 1。

-- 查出员工工资最高的前三名并按顺序显示序号
select rownum,e2.* from (select * from emp e1 order by sal desc) e2 where rownum<4;

例(rownum):查询出员工表中工资最高的前三名

-- 查询出薪水大于本部门平均薪水的员工
select * from emp e1,(select deptno,avg(sal) avgsal from emp group by deptno) s1 where e1.deptno=s1.deptno and e1.sal>s1.avgsal

例(avg):查询出薪水大于本部门平均薪水的员工

--例(sum):统计每年入职的员工个数
select to_char(hiredate,'yyyy') year,count(1) count from emp group by to_char(hiredate,'yyyy');

select
sum(decode(year,'',count)) "1987",
sum(decode(year,'',count)) "1980",
sum(decode(year,'',count)) "1981",
sum(decode(year,'',count)) "1982",
sum(count) Total
from
(select to_char(hiredate,'yyyy') year,count(1) count from emp group by to_char(hiredate,'yyyy')) r1;

例(sum):统计每年入职的员工个数

-- 方式一
delete from customer where rowid not in (select min(rowid) from customer group by cname)
-- 方式二
delete from customer c1 where rowid >(select min(rowid) from customer c2 where c1.cname=c2.cname)

例(rowid):删除表中重复的数据

select * from (select rownum hanghao,e1.* from emp e1) e2 where e2.hanghao between 1 and 3;

例(rownum):分页查询

Oracle(2)之多表查询&子查询&集合运算的更多相关文章

  1. oracle 基础SQL语句 多表查询 子查询 分页查询 合并查询 分组查询 group by having order by

    select语句学习 . 创建表 create table user(user varchar2(20), id int); . 查看执行某条命令花费的时间 set timing on: . 查看表的 ...

  2. sql:除非另外还指定了 TOP 或 FOR XML,否则,ORDER BY 子句在视图、内联函数、派生表、子查询

    执行sql语句: select * from ( select * from tab where ID>20 order by userID desc ) as a order by date ...

  3. Python-select 关键字 多表查询 子查询

    sql 最核心的查询语句!!!! 增删改 单表查询 select语句的完整写法 关键字的书写顺序 执行顺序 多表查询 笛卡尔积 内连接 左外连接 右外连接 全外连接 通过合并左外连接和右外连接 子查询 ...

  4. ORDER BY 子句在视 图、内联函数、派生表、子查询和公用表表达式中无效

    SQL语句: select * from (select distinct t2.issue,cashmoney from (select distinct issue from lot_gamepa ...

  5. [转]sql:除非另外还指定了 TOP 或 FOR XML,否则,ORDER BY 子句在视图、内联函数、派生表、子查询

    执行sql语句: select * from ( select * from tab where ID>20 order by userID desc ) as a order by date ...

  6. [sql Server]除非另外还指定了TOP 或 FOR XML,否则,ORDER BY 子句在视图、内联函数、派生表、子查询和公用表表达式中无效

    今天遇到一个奇怪的问题,项目突然要从mysql切换到sql server数据库,包含order by 子句的嵌套子查询报错. 示例:select top 10 name,age,sex from ( ...

  7. ylb: SQL表的高级查询-子查询

    ylbtech-SQL Server: SQL Server- SQL表的高级查询-子查询 SQL Server 表的高级查询-子查询. 1,ylb:表的高级查询-子查询返回顶部 --======== ...

  8. Oracle的查询-子查询

    --子查询 --子查询返回一个值 --查询出工资和scott一样的员工信息 select * from emp where sal in (select sal from emp where enam ...

  9. mysql中【update/Delete】update中无法用基于被更新表的子查询,You can't specify target table 'test1' for update in FROM clause.

    关键词:mysql update,mysql delete update中无法用基于被更新表的子查询,You can't specify target table 'test1' for update ...

  10. 【数据库】SQL经典面试题 - 数据库查询 - 子查询应用二

    上节课我们通过子查询,完成了查询的最高分学生的需求,今天我们来学习子查询的分类,以及通过子查询来完成工作中经常遇到一些个性化需求. 子查询概念: 一个SELECT语句嵌套在另一个SELECT语句中,子 ...

随机推荐

  1. 单元测试以文件流的形势传参调用api进行测试

    [TestMethod] public void Test() { byte[] buffer;//文件转换为二进制流 string path = @"E:\aaa"; Encod ...

  2. Excel 函数集(使用过的)

    1.   SUBTOTAL函数  筛选结果求和 SUBTOTAL(函数编号, 区域) 函数编号 为 1 到 11(包含隐藏值)或 101 到 111(忽略隐藏值)之间的数字,指定使用何种函数在数据清单 ...

  3. 【CF679D】Bear and Chase 最短路+乱搞

    [CF679D]Bear and Chase 题意:近日,鼠国的头号通缉犯,神出鬼没的怪盗——Joker正于摩登市出没!对于名侦探Jack来说,这正是将其捉拿归案的大号时机.形式化地,摩登市可以看成一 ...

  4. Linux 更新vim

    https://blog.csdn.net/linuxnews/article/details/52938583 https://blog.csdn.net/nzyalj/article/detail ...

  5. Win 10 计算机管理失效(Windows找不到文件“C:\ProgramData\Microsoft\Windows\Start Menu\Programs\Administrative Tools\Computer Management.lnk)

    Windows找不到文件“C:\ProgramData\Microsoft\Windows\Start Menu\Programs\Administrative Tools\Computer Mana ...

  6. ThinkPHP框架 系统规定的方法查询数据库内容!!同时也支持原生的SQL语句!

    <?php namespace Admin\Controller; use Think\Controller; class MainController extends Controller{ ...

  7. CentOs安装和使用

    ● 去掉图形界面 http://blog.csdn.net/op_zoro/article/details/44993881 ● centos 7覆盖windows vi /boot/grub2/gr ...

  8. ubuntu16.04安装kinetic调用gazebo_control解决方案

    解决方案 sudo apt-get install ros-kinetic-gazebo-ros-control

  9. centos 安装oracle 报Checking swap space: 0 MB available, 150 MB required. Failed <<<<

    1 系统环境 centos 6.5 oracle 11g 内存 16G 硬盘 ssd 250G 2 运行安装命令: [oracle@localhost database]$ ./runInstalle ...

  10. URL地址中的#

    1.#的涵义 代表网页中的一个位置.井号后面的字符,就是该位置的标识符.比如, http://www.baidu.com/index.html#one 就代表网页index.html的one位置.浏览 ...