Oracle子查询之高级子查询
Oracle 高级子查询
高级子查询相对于简单子查询来说,返回的数据行不再是一列,而是多列数据。
1,多列子查询
主查询与子查询返回的多个列进行比较
查询与141号或174号员工的manager_id和department_id相同的其他员工的employee_id, manager_id, department_id
select employee_id, manager_id, department_id
from employees
where (manager_id,department_id) in ( --由于内查询返回两列,所以此处也必须使用两列来进行对应,并且列的顺序一致
select manager_id,department_id --内查询返回2列
from employees
where employee_id in (141,174)
)
and employee_id not in (141,174);
2,FROM子句中使用子查询
将子查询的结果集,作为一个虚表,然后从这个虚表中查询数据行
返回比本部门平均工资高的员工的last_name, department_id, salary及平均工资
select last_name, e.department_id, salary,avgsal
from employees e,(select department_id,round(avg(salary)) avgsal
from employees
group by department_id) avg_sal
where e.department_id = avg_sal.department_id
and e.salary > avg_sal.avgsal
order by e.department_id;
3,单列子查询的其他应用
在主查询的select列表中使用
问题:显式员工的employee_id,last_name和location。其中,若员工department_id与location_id为1800的department_id相同,
则location为’Canada’,其余则为’USA’。
select employee_id,last_name,department_id,
(case when department_id = (select department_id
from departments
where location_id = 1800
)
then 'USA'
else 'Canada' end) location
from employees
order by department_id;
或
select employee_id,last_name,department_id,
(case department_id when (select department_id
from departments
where location_id = 1800
)
then 'USA'
else 'Canada' end) location
from employees
order by department_id;
或
select employee_id,last_name,department_id,
decode(department_id,(select department_id
from departments
where location_id = 1800
),'USA',
'Canada') location
from employees
order by department_id;
4,相关子查询,
子查询语句中使用了主查询语句中的表中的数据(这个数据不一定在主查询的select语句中)
相关子查询按照一行接一行的顺序执行,即子查询取主查询表中的每一行数据值,主查询的每一行都执行一次子查询。
使用相关子查询,要考虑是否必须使用、是否合理,否则在不需要使用相关子查询的情况下就能轻易得到查询结果,使用相关子查询反而会使查询的效率下降。
用法:
get:从主查询所用的表中获取候选列
execute:子查询使用主查询的数据,并进行相关的筛选
use:如果满足内查询的条件则返回值
例1:查询员工的employee_id,last_name,要求按照员工的department_name排序
select employee_id,last_name,e.department_id,department_name
from employees e,departments dd
where e.department_id = dd.department_id(+) --该查询返回employees表中的所有行,没返回一行,都会取一个employee_id到子查询中
order by (select department_name
from departments d
where e.department_id = d.department_id); --子查询根据主查询返回的employee_id,在departments表中进行查找,有则返回数据。
desc;
例2:查询员工中工资大于本部门平均工资的员工的last_name,salary,department_id和本部门的平均工资
select last_name,salary,e1.department_id,ss.avgsal
from employees e1,(select department_id,round(avg(salary)) avgsal
from employees
group by department_id) ss --该from字句同前例
where e1.department_id = ss.department_id
and salary > (
select round(avg(salary))
from employees e2
where e1.department_id = e2.department_id --将外查询中的department_id,查找该处的department_id,并将所有的返回结果分组
group by e2.department_id)
order by e1.department_id;
例3:若employees表中employee_id与job_history表中employee_id相同且job_history表中employee_id的数目不小于2,
输出employees中这些相同id的员工的employee_id,last_name和其job_id
select e1.employee_id ,last_name,e1.job_id
from employees e1
where (select count(job_id)
from job_history j1
where e1.employee_id = j1.employee_id) >= 2
5,exists
EXISTS 操作符检查在子查询中是否存在满足条件的行
如果在子查询中存在满足条件的行:
不在子查询中继续查找
条件返回 TRUE
如果在子查询中不存在满足条件的行:
条件返回 FALSE
继续在子查询中查找
查询公司管理者的employee_id,last_name,job_id,department_id信息
子查询
select employee_id,last_name,job_id,department_id
from employees e1
where e1.employee_id in (
select distinct(manager_id)
from employees);
自连接
select distinct e1.employee_id,e1.last_name,e1.job_id,e1.department_id
from employees e1,employees e2
where e1.employee_id = e2.manager_id;
相关子查询
select employee_id,last_name,job_id,department_id
from employees e1
where e1.employee_id in (
select manager_id
from employees e2
where e1.employee_id = e2.manager_id);
使用exists相关子查询
select employee_id,last_name,job_id,department_id
from employees e1
where exists (
select 'A' --根据e1表中的每个 employee_id在e2中查找 manager_id,如果找到,子查询返回true,主查询返回该数据行
from employees e2
where e1.employee_id = e2.manager_id);
6.not exists
查询departments表中,不存在于employees表中的部门的department_id和department_name
select d.department_id,d.department_name
from departments d
where not exists ( --not exists 返回子查询中false的结果
select 'X' --根据e1表中的每个 employee_id在e2中查找 manager_id,如果找到,子查询返回true,即该部门在departments中存在,也有员工
from employees e
where d.department_id = e.department_id);
7.相关更新
使用相关子查询依据一个表中的数据更新另一个表的数据
例:向employees中添加一列 department_name ,并更具department_id填充
update employees e1
set department_name = (
select department_name
from departments
where e1.department_id = department_id);
8,相关删除
使用相关子查询依据一个表中的数据删除另一个表的数据
例:删除表employees中,其与emp_history表皆有的数据
delete from employees e1
where employee_id in (
select employee_id
from job_history
where e1.employee_id = employee_id);
9,使用 WITH 子句
可以避免在 SELECT 语句中重复书写相同的语句块
WITH 子句将该子句中的语句块执行一次并存储到用户的临时表空间中
使用 WITH 子句可以提高查询效率
例,查询公司中各部门的总工资,大于公司中各部门的平均总工资,的部门信息
使用普通方法:使用普通方法,汇总后的语句冗长不易理解
--各部门总工资
select department_id,sum(salary) sum_sal from employees group by department_id;
--各部门的平均总工资
select sum(to_sal.sum_sal)/count(*) to_avg_sal
from (select department_id,sum(salary) sum_sal
from employees group by department_id) to_sal;
--根据前面两项,查询最终结果
select to_sal.department_id,sum_sal
from (select department_id,sum(salary) sum_sal from employees group by department_id) to_sal,
(select sum(to_sal.sum_sal)/count(*) to_avg_sal from (select department_id,sum(salary) sum_sal from employees group by department_id) to_sal) avg_sal
where to_sal.sum_sal > avg_sal.to_avg_sal;
使用with字句
with to_sal as --各部门总工资
(select department_id,sum(salary) sum_sal from employees group by department_id),
avg_sal as --各部门的平均总工资
(select sum(to_sal.sum_sal)/count(*) to_avg_sal from to_sal)
select department_id,sum_sal
from to_sal
where to_sal.sum_sal > (
select avg_sal.to_avg_sal
from avg_sal);
with
to_sal as
(select department_id,sum(salary) sum_sal from employees group by department_id),
avg_sal as
(select sum(to_sal.sum_sal)/count(*) to_avg_sal from to_sal)
select department_id,sum_sal,to_avg_sal
from to_sal,avg_sal
where to_sal.sum_sal > avg_sal.to_avg_sal;
注明:本博文系学习尚硅谷网易云课堂教学课程总结输出。
Oracle子查询之高级子查询的更多相关文章
- Oracle学习DayFour(高级子查询)
一.高级子查询 1.多列子查询 定义:主查询与子查询返回的多个列进行比较 多列子查询中的比较分为两种:成对比较:不成对比较 实例:查询与141号或174号员工的manager_id和departmen ...
- Oracle 学习笔记 14 -- 集合操作和高级子查询
Oracel提供了三种类型的集合操作:各自是并(UNION) .交(INTERSECT). 差(MINUS) UNION :将多个操作的结果合并到一个查询结果中,返回查询结果的并集,自己主动去掉反复的 ...
- Oracle入门第六天(下)——高级子查询
一.概述 主要内容: 二.子查询介绍 1.简单子查询(WHERE子查询) SELECT last_name FROM employees WHERE salary > (SELECT salar ...
- Oracle系列八 高级子查询
子查询 子查询 (内查询) 在主查询执行之前执行 主查询(外查询)使用子查询的结果 多列子查询 主查询与子查询返回的多个列进行比较 多列子查询中的比较分为两种: 成对比较 问题:查询与141号或174 ...
- Oracle-3 - :超级适合初学者的入门级笔记--用户权限,set运算符,高级子查询
上一篇的内容在这里第二篇内容, 用户权限:创建用户,创建角色,使用grant 和 revoke 语句赋予和回收权限,创建数据库联接 创建用户:create user xxx identified b ...
- Oracle子查询之简单子查询
Oracle 简单子查询 顾名思义,简单子查询是嵌套在 SQL 语句中的另一个SELECT 语句,并且子查询只返回一列数据 1,单行子查询: 子查询 (内查询) 在主查询之前一次执行完成.子查询的结果 ...
- ORACLE 多表连接与子查询
Oracle表连接 SQL/Oracle使用表连接从多个表中查询数据 语法格式: select 字段列表from table1,table2where table1.column1=table2.co ...
- Oracle子查询和多表查询
多表查询需要用到表的连接 连接可以分为:(自行百度) 交叉连接(数字逻辑的笛卡尔积,不做解释) 等值连接 例如:select * from t_a, t_b where t_a.xx = t_b.xx ...
- Oracle 数据库基础学习 (六) 子查询
子查询在一个select中出现多个嵌套查询语句 1.在where子句中使用子查询(一般返回"单行单列" "单行多列" "多行单列"(可以提供 ...
随机推荐
- JS判断数字类型
JavaScript判断输入是否为数字类型的方法总结 前言 很多时候需要判断一个输入是否位数字,下面简单列举集中方法. 第一种方法 isNaN isNaN 返回一个 Boolean 值,指明提供的值是 ...
- 空白符对HTML结构的影响与解决方案
何为空白符? 空白符: 空格.制表符.换行符 注意:浏览器在解析HTML时会把所有空白符合并成一个空格 空白符对HTML结构的影响 HTML5中<textarea>标签placeholde ...
- php接收post过来的json数据
<html> <head> <title>json</title> <script src="//cdn.bootcss.com/jqu ...
- ArcGIS软件操作——地图配准
初次写博文,出现措词不当.表述不明确等之类的问题,敬请见谅,但会努力做好.同时,也欢迎各位提出意见,共同交流,共同进步! 直奔主题——运用ArcGIS软件对地图进行配准! 1 数据准备:网络下载的中国 ...
- 购物车动画(Android)
购物车动画(Android) 前言:当我们写商城类的项目的时候,一般都会有加入购物车的功能,加入购物车的时候会有一些抛物线动画,最近做到这个功能,借助别人的demo写了一个. 效果: 开发环境:And ...
- 致 BitClub 矿池,你们为什么要对比特币网络发动交易延展性攻击?
原文:https://medium.com/@bithernet/bitclub-why-are-you-doing-malleability-attack-now-6faa194b2146#.v4y ...
- webstorm中使用git
webstorm中使用git将代码放入tfs两种方式: 直接在tfs上建立仓库,复制仓库地址,然后在本地打开webstorm,然后git克隆这个仓库 使用git命令将本地项目上传到tfs git re ...
- java Maven项目右键没有maven菜单项的解决方案!
修改项目.project文件,确保有maven2Builder和maven2Nature2个标签: <?xml version="1.0" encoding="UT ...
- 杨氏矩阵C++实现
何为杨氏矩阵?这个网上的介绍很多,下面给出杨氏矩阵搜索算法: #include <iostream> using namespace std; // 杨氏矩阵查找算法 ], int N, ...
- iframe加载顺序导致数据访问出现问题
背景: 一个页面A内有一个iframe,src指向了B页面. 问题: 页面A通过Ajax获取服务器数据,并赋值给了页面A的全局变量gData,页面B要用到页面A的数据gData.那么问题来了当B访问g ...