九、高级查询(分组,子查询)
查询升级版:
需要用到三张表
员工表:
desc emp
EMPNO 员工号
ENAME 员工姓名
JOB 员工职位
MGR 老板员工号
HIREDATE 员工入职日期
SAL 员工月薪
COMM 员工奖金
DEPTNO 员工所在部门部门号

查看当前用户:
show users;
select * from emp;
部门表:
desc dept
deptno 部门号
dname 部门名称
LoC 部门地点
select * from dept;
工资级别表:
desc salgrade
GRADE 级别
LOSAL 级别最低薪水
HISAL 级别最高薪水
select * from salgrade;

十、分组查询
1、分组函数的概念
分组函数作用于一组数据,并对一组数据返回一个值

2、分组函数的使用
常用的分组函数:AVG、SUM、MIN、MAX、COUNT、WM_CONCAT(行转列)
(1)求出员工的平均工资和工资总额
select avg(SAL) as 平均工资,sum(SAL) as 工资总额 from emp;
(2)求出员工工资的最大值和最小值
select max(SAL) as 工资的最大值,min(SAL) as 工资的最小值 from emp;
(3)求出员工的总人数
select COUNT(*) as 总人数 from emp;
(4)查询部门数(去除重复)
select COUNT(distinct deptno) as 部门数 from emp;
(5)行转列
设置行宽
set linesize 200
col 部门中员工的姓名 for a60
select deptno as 部门号,wm_concat(ename) as 部门中员工的姓名 from emp group by deptno;
(6)分组函数与空值
a、统计员工的平均工资(三种方法统计薪水)
select sum(sal)/count(*) 一,sum(sal)/count(sal) 二, avg(sal) 三 from emp;
b、统计员工的平均奖金(三个值不相同)
select sum(comm)/count(*) 一,sum(comm)/count(comm) 二, avg(comm) 三 from emp;

注意:
分组函数会自动忽略空值
(7)在分组函数中使用NVl函数
注意:NVL函数使分组函数无法忽略空值
改写b
所有员工都统计
select sum(comm)/count(*) 一,sum(comm)/count(nvl(comm,0)) 二, avg(nvl(comm,0)) 三 from emp;
非空员工的统计
select sum(comm)/count(comm) 二, avg(comm) 三 from emp;

3、使用GROUP BY 子句数据分组
(1)求出employee表中各部门的平均工资,并按照部门号分组
select deptno 部门号,avg(sal) 平均工资 from emp group by deptno;
注意:
在select列表中所有未包含在组函数中的列都应该包含在group by子句中
就是select 列名一,列名二 from emp group by 列名一;
因为列名一不是列名二的内容,所有要加在group by 后面
(2)求每个部门的平均工资,要求显示:部门的平均工资
select avg(sal) 平均工资 from emp group by deptno;

注意:
包含在group by 子句中的列不必包含在select列表中
(3)使用多个列分组
查询员工的薪水总额
select deptno 员工号,job 工作,sum(sal) 总薪水 from emp group by deptno,job order by deptno;
修改语句:
edit
编辑后保存
执行语句:
/
(4)非法使用组函数
所包含与select列表中,而未包含于组函数中的列都必须包含于group by 子句中
经典错误
select deptno,count(ename) from emp;
正确形式:
select deptno,count(ename) from emp group by deptno;

4、使用HAVING子句过滤分组结果集
(1)求平均工资大于2000的部门
select deptno 部门号,avg(sal) 平均工资 from emp group by deptno having avg(sal)>2000;

where与having的区别
a、不能在where子句中使用组函数
b、可以在having子句中使用组函数

(2)查询10号部门的平均工资(where和having可以通用的情况)
select deptno 部门号,avg(sal) 平均工资 from emp group by deptno having deptno=10;
select deptno 部门号,avg(sal) 平均工资 from emp where deptno=10 group by deptno;

注意:
从SQL优化的角度看,尽量使用where,where使得分组记录数大大降低,从而提高效率

5、在分组查询中使用order by子句
(1)求每个部门的平均工资,要求显示:部门号,部门的平均工资,并且按照工资升序排列(可以按照:列、别名、表达式、序号进行排序)
select deptno 部门号,avg(sal) 平均工资 from emp group by deptno order by avg(sal);
写法二:
select deptno 部门号,avg(sal) 平均工资 from emp group by deptno order by 2;
追加降序命令:
a(后接两个或以上空格)
a desc
/
select deptno 部门号,avg(sal) 平均工资 from emp group by deptno order by 2 desc;

(2)分组函数的嵌套
求部门平均工资的最大值
思路:(a)通过avg函数求出每个部门的平均工资
(b)嵌套max函数求出部门平均工资的最大值
select max(avg(sal)) 平均工资 from emp group by deptno;

6、group by语句的增强
(1)按部门、不同的职位、统计工资总额,
按部门,统计工资总额
统计工资总额
select deptno 员工号,job 工作,sum(sal) 总薪水 from emp group by deptno,job order by deptno;
select deptno 员工号 sum(sal) 总薪水 from emp group by deptno
select sum(sal) 总薪水 from emp
上三句相加得下边一句话
break on deptno skip 2
set pagesize 30
select deptno,job,sum(sal) from emp group by rollup(deptno,job);(不可设置别名)

7、SQL*PLus的报表功能
报表包括:标题/页码/别名等
ttitle col 15 '我的报表' col 35 sql.pno 空15个列显示我的报表这个标题,空35个列
col deptno heading 部门号 标题设置为不猛号
col job heading 职位
col sum(sal) heading 工资总额
break on deptno skip 1
select deptno,job,sum(sal) from emp group by rollup(deptno,job);
保存一个目录中,get 文件路径
@路径执行

8、多表查询
(1)什么是多表查询?
从多个表中获取数据,就是多表查询
(2)迪卡尔集
例子演示:
create table dept_disk(
deptno number(5,0) primary key,
dname varchar2(20)
);
insert into dept_disk values(1,'开发部');
insert into dept_disk values(2,'行政部');
create table emp_disk(
emptno number(5,0),
ename varchar2(20),
sal number(5,0),
deptno number(5,0)
);
insert into emp_disk values(1,'开发部');
insert into emp_disk values(2,'行政部');
两包张表按照如下规则合并成一张表,叫做迪卡尔集表
(1)迪卡尔集表的列数,等于每张表的列数相加,迪卡尔集表的行数,等于每张表的行数相乘.得到笛卡尔全集
(2)多表查询的核心是连接条件.连接条件的个数,表的个数减一,两张表至少有一个连接条件.
注意:
(a)为了避免迪卡尔集,可以在where加入有效的连接条件.
(b)在实际运行环境下,应避免使用笛卡尔全集.
连接类型:
(3)等值连接
连接条件是个等号,就是等值连接
例子:查询员工信息,要求显示:员工号,姓名,月薪,部门名称
select e.empno as 员工号,e.ename as 姓名,e.sal as 月薪,d.dname as 部门名称 from emp e, dept d where e.deptno=d.deptno;
(4)不等值连接
例子:查询员工信息,要求显示:员工号,姓名,月薪,薪水等级(用between...and需要小值在前,大值在后)
select e.empno as 员工号,e.ename as 姓名,e.sal as 月薪, s.grade as 薪水等级 from emp e,salgrade s where e.sal between s.losal and s.hisal;
(5)外连接
例子:按部门统计员工人数,要求显示:部门号,部门名称,人数
select d.deptno as 部门号,d.dname as 部门名称,count(e.empno) as 人数 from dept d, emp e where e.deptno(+)=d.deptno group by d.deptno,d.dname;

注意:
(a)核心:通过外连接,把对于连接条件不成立的记录,仍然包含在最后的结果中
(b)左外连接:当条件不成立的时候,等号左边的表仍然被包含(等号右边加(+))
(c)右外连接:当连接条件不成立的时候,等号右边的表仍然被包含(等号左边加(+))

(6)自连接
例子:查询员工姓名和员工的老板姓名
连接条件员工的老板号等于老板的员工号
select e.ename as 员工姓名,b.ename as 老板姓名 from emp e,emp b where e.mgr=b.empno;

注意:
(a)核心:通过别名,将同一张表视为多张表

自连接存在的问题:
(1)自连接不做任何操作,得到的就是笛卡尔乘积
select count(*) from emp e,emp b;
(a)问题一:自连接不是和操作大表(因为自乘会有很大的数据)
解决方法:层次查询
说明:某些情况下,可以替代自连接,本质上是一个单表查询
(7)层次查询
原理:关系成成树状图
select level as 层次,empno as 员工号,ename as 员工姓名,sal as 月薪,mgr as 老板号 from emp connect by prior empno=mgr start with mgr is null order by 1;
优点:单表查询
缺点:没有自连接直观

Oracle学习笔记(七)的更多相关文章

  1. Oracle学习笔记七 锁

    锁的概念 锁是数据库用来控制共享资源并发访问的机制. 锁用于保护正在被修改的数据 直到提交或回滚了事务之后,其他用户才可以更新数据 对数据的并发控制,保证一致性.完整性.

  2. Oracle学习笔记三 SQL命令

    SQL简介 SQL 支持下列类别的命令: 1.数据定义语言(DDL) 2.数据操纵语言(DML) 3.事务控制语言(TCL) 4.数据控制语言(DCL)  

  3. (转)Qt Model/View 学习笔记 (七)——Delegate类

    Qt Model/View 学习笔记 (七) Delegate  类 概念 与MVC模式不同,model/view结构没有用于与用户交互的完全独立的组件.一般来讲, view负责把数据展示 给用户,也 ...

  4. Learning ROS for Robotics Programming Second Edition学习笔记(七) indigo PCL xtion pro live

    中文译著已经出版,详情请参考:http://blog.csdn.net/ZhangRelay/article/category/6506865 Learning ROS forRobotics Pro ...

  5. oracle学习笔记第一天

    oracle学习笔记第一天 --oracle学习的第一天 --一.几个基础的关键字   1.select select (挑选) 挑选出显示的--列--(可以多列,用“,”隔开,*表示所有列),为一条 ...

  6. Typescript 学习笔记七:泛型

    中文网:https://www.tslang.cn/ 官网:http://www.typescriptlang.org/ 目录: Typescript 学习笔记一:介绍.安装.编译 Typescrip ...

  7. python3.4学习笔记(七) 学习网站博客推荐

    python3.4学习笔记(七) 学习网站博客推荐 深入 Python 3http://sebug.net/paper/books/dive-into-python3/<深入 Python 3& ...

  8. Go语言学习笔记七: 函数

    Go语言学习笔记七: 函数 Go语言有函数还有方法,神奇不.这有点像python了. 函数定义 func function_name( [parameter list] ) [return_types ...

  9. Oracle学习笔记——点滴汇总

    Oracle学习笔记——点滴汇总 http://www.botangdb.com/ Oracle GI = Grid Infrastructure = ASM + Cluster

随机推荐

  1. eclipse中点不出来提示

    当在用eclipse或是myeclipse时,可能会遇到不能自动提示,就是当你用到点的时候,后面不会出现相关的提示信息.这时,解决方法如下 : 1.菜单window->Preferences-& ...

  2. FastAdmin 是如何利用 Git 管理插件代码的?

    FastAdmin 是如何利用 Git 管理插件代码的? 由于 FastAdmin 的插件很多,如果每一个插件用一个项目来管理,可以倒是可以,但是项目还多了. 但是如果使用文件夹在同一级的的方式又不方 ...

  3. java 方法(函数)

    所谓方法,就是用来解决一类问题的代码的有序组合,是一个功能模块. 一般情况下,定义一个方法的语法是: 其中: 1. 访问修饰符:方法允许被访问的权限范围, 可以是 public.protected.p ...

  4. 收藏的一些github开源项目,在这里记录一下

    1.在windows系统上在命令行中执行rm -fr */*.file 会报错, rimraf这个项目提供了跨平台支持rm命令,github地址: https://github.com/isaacs/ ...

  5. Linux虚拟机与Windows共享文件

    1.找到“虚拟机” >>> “设置” 2.选项 >>> 共享文件夹 >>>  总是启用 >>> 添加,选择你想要共享的文件. 注 ...

  6. php soap实例讲解

    一,什么是soap,什么是wsdl,为什么要用他们 SOAP是基于XML和HTTP通信协议,xml各种平台,各种语言都支持的一个种语言.http呢它得到了所有的因特网浏览器及服务器的支持. WSDL ...

  7. mangle和demangle

    转:https://www.cnblogs.com/robinex/p/7892795.html. mangle和demangle C/C++语言在编译以后,函数的名字会被编译器修改,改成编译器内部的 ...

  8. Django UrL 解析

    Django的路由系统 URLconf 本质是URL与要为该URL调用的视图函数之间的映射表:你就是以这种方式告诉Django,对于客户端发来的某个URL调用哪一段逻辑代码对应执行. 1.1 djan ...

  9. Android 多线程注意事项

    参考:http://blog.csdn.net/x86android/article/details/14161981 http://geeksun.iteye.com/blog/1447708 An ...

  10. Python web框架——Tornado

    Tornado是一个Python Web框架和异步网络库,最初由FriendFeed开发.通过使用非阻塞网络I / O,Tornado可以扩展到数万个开放连接,使其成为需要长时间连接每个用户的长轮询, ...