Oracle(4)

  1. 子查询

子查询语法很简单,就是select 语句的嵌套使用。

查询工资比SCOTT高的员工信息

分析:两步即可完成

1. 查出SCOTT的工资 SQL> select ename, sal from emp where ename='SCOTT'   其工资3000

2. 查询比3000高的员工 SQL> select * from emp where sal>3000  

通过两步可以将问题结果得到。子查询,可以将两步合成一步。

——子查询解决的问题:问题本身不能一步求解的情况。

SQL> select *

  from emp  

  where sal  >  (select sal  

 from emp  

 where ename='SCOTT')  

子查询语法格式:

SELECT select_list

FROM table

WHERE expr operator

  (SELECT select_list

     FROM table);

本章学习目标: 

描述子查询可以解决的问题

定义子查询(子查询的语法)

列出子查询的类型。

书写单行子查询和多行子查询。

定义子查询 需要注意的问题

1. 合理的书写风格 (如上例,当写一个较复杂的子查询的时候,要合理的添加换行、缩进)

2. 小括号( ) 

3. 主查询和子查询可以是不同表,只要子查询返回的结果主查询可以使用即可

4. 可以在主查询的where、select、having、from后都可以放置子查询

5. 不可以在主查询的group by后面放置子查询 (SQL语句的语法规范)

6. 强调:在from后面放置的子查询(***) from后面放置是一个集合(表、查询结果)

7. 一般先执行子查询(内查询),再执行主查询(外查询);但是相关子查询除外  

8. 一般不在子查询中使用order by, 但在Top-N分析问题中,必须使用order by  

9. 单行子查询只能使用单行操作符;多行子查询只能使用多行操作符

10. 子查询中的null值

主、子查询在不同表间进行。

查询部门名称是“SALES”的员工信息

主查询:查询员工信息。select * from emp;

子查询:负责得到部门名称(在dept表中)、部门号对应关系。select deptno from dept where dname='SALES'    

SQL> select *

 from emp  

   where deptno= (select deptno  

from dept

where dname='SALES')    

主查询,查询的是员工表emp,子查询,查询的是部门表dept。是两张不同的表。

将该问题使用“多表查询”解决:

SQL> select e.*  from emp e, dept d  where e.deptno=d.deptno and d.dname='SALES'    

两种方式哪种好呢?

※SQL优化: 理论上,既可以使用子查询,也可以使用多表查询,尽量使用“多表查询”。子查询有2次from

不同数据库处理数据的方式不尽相同,如Oracle数据库中,子查询地位比较重要,做了深入的优化。有可能实际看到结果是子查询快于多表查询。

在主查询的where select having from 放置子查询

子查询可以放在select后,但,要求该子查询必须是 单行子查询:(该子查询本身只返回一条记录,2+叫多行子查询)

SQL> select empno, ename, (select dname from dept where deptno=10) 部门 from emp   

注意:SQL中没有where是不可以的,那样是多行子查询。

进一步理解查询语句实际上是在表或集合中通过列名来得到行数据,子查询如果是多行,select无法做到这一点。

在 having 后 和 where 类似。但需注意在where后面不能使用组函数。       

​​​​​​​在from后面放置的子查询(***)

代表一个数据集合查询结果(SQL)语句本身也代表一个集合

查询员工的姓名、薪水和年薪:

说明:该问题不用子查询也可以完成。但如果是一道填空题:select * from ___________________

因为显示的告诉了,要使用select *

SQL> select * from (select ename, sal, sal*12 年薪 from emp);

将select 语句放置到from后面,表示将select语句的结果,当成表来看待。这种查询方式在Oracle语句中使用比较频繁。

单行子查询只能使用单行操作符;多行子查询只能使用多行操作符

  1. 单行子查询:

单行子查询就是该条子查询执行结束时,只返回一条记录(一行数据)。

使用单行操作符:

=、>、>=、<、<=、<>

如:

SELECT last_name, job_id, salary

FROM   employees

WHERE  job_id =  

                (SELECT job_id

                 FROM   employees

                 WHERE  employee_id = 141)

AND    salary >

                (SELECT salary

                 FROM   employees

                 WHERE  employee_id = 143);

再如:

SELECT last_name, job_id, salary

FROM   employees

WHERE  salary =

                (SELECT MIN(salary)

                 FROM   employees);

也可以在having子句中使用:

SELECT   department_id, MIN(salary)

FROM     employees

GROUP BY department_id

HAVING   MIN(salary) >

                       (SELECT MIN(salary)

                        FROM   employees

                        WHERE  department_id = 50);

上面的例子告诉我们:

1. 单行子查询,只能使用单行操作符(=号、>号)

2. 在一个主查询中可以有多个子查询。

3. 子查询里面可以嵌套多层子查询。

4. 子查询也可以使用组函数。子查询也是查询语句,适用于前面所有知识。

非法使用子查询的例子:

SELECT employee_id, last_name

FROM   employees

WHERE  salary =

                (SELECT   MIN(salary)

                 FROM     employees

                 GROUP BY department_id);

在此例中,单行操作符“=”连接了返回多条记录的子查询。查询语句执行会报错。

多行子查询:

子查询返回2条记录以上就叫多行。

多行操作符有:

IN 等于列表中的任意一个

ANY 和子查询返回的任意一个值比较

ALL 和子查询返回的所有值比较

IN(表示在集合中)

查询部门名称为SALES和ACCOUNTING的员工信息。

分析:部门名称在dept表中,员工信息在emp表中。→

  子查询应先去dept表中将SALES和ACCOUNTING的部门号得到,交给主查询得员工信息

SQL> select *

  from emp

  where deptno in (select deptno

  from dept

  where dname= 'SALES 'or dname= 'ACCOUNTING ');  

使用 多表查询 来解决该问题:

SQL> select e.*

  from emp e, dept d

  where  e.deptno=d.deptno and (d.dname= 'SALES ' or d.dname= 'ACCOUNTING ');

这种解决方式,注意使用()来控制优先级。 

如果查询不是这两个部门的员工,只要把in → not in就可以了,注意不能含有空值。

ANY(表示和集合中的任意一个值比较)

查询薪水比30号部门任意一个员工高的员工信息:

分析:首先查出30号部门的员工薪水的集合,然后 > 它就得到了该员工信息。

SQL> select * from emp where sal > (select sal from emp where deptno=30);   正确吗?

这样是错的,子句返回多行结果。而‘>’是单行操作符。 ——应该将‘>’替换成‘> any’

实际上>集合的任意一个值,就是大于集合的最小值。

若将这条语句改写成单行子查询应该怎么写呢?

SQL> select * from emp where sal > (select min(sal) from emp where deptno=30);  

ALL(表示和集合中的所有值比较):

查询薪水比30号部门所有员工高的员工信息。

SQL> Select * from emp where sal > all (select sal from emp where deptno=30);  

同样,将该题改写成单行子句查询:

SQL> Select * from emp where sal > (select max(sal) from emp where deptno=30);  

对于any 和 all 来说,究竟取最大值还是取最小值,不一定。将上面的两个例子中的“高”换成“低”,any和all就各自取相反的值了。

子查询中null 

判断一个值等于、不等于空,不能使用=和!=号,而应该使用is 和 not。

如果集合中有NULL值,不能使用not in。如: not in (10, 20, NULL),但是可以使用in为什么呢

先看一个例子:

查询不是老板的员工信息:

分析:不是老板就是树上的叶子节点。在emp表中有列mgr,该列表示该员工的老板的员工号是多少。那么,如果一个员工的员工号在这列中,那么说明这员工是老板,如果不在,说明他不是老板。

SQL> select * from emp where empno not in (select mgr from emp);  但是运行没有结果,因为有NULL 

查询是老板的员工信息: 只需要将not去掉。

SQL> select * from emp where empno in (select mgr from emp );  

还是我们之前null的结论:in (10, 20, null) 可以,not in (10, 20, null) 不可以

例如:a not in(10, 20, NULL) 等价于 (a != 10) and (a != 20) and (a != NULL)

  因为,not in操作符等价于 !=All,最后一个表达式为假,整体假。如:

SELECT emp.last_name

FROM   employees emp

WHERE  emp.employee_id NOT IN

                             (SELECT mgr.manager_id

                              FROM  employees mgr);

  a in (10, 20, NULL)   等价于 (a = 10) or (a = 20) or (a = null)只要有一个为真即为真。

  in 操作符等价于 = Any

所以子查询中,如果有NULL值,主查询使用where xxx=子查询结果集。永远为假。

继续,查询不是老板的员工信息。 只要将空值去掉即可。

SQL>  select * from emp where empno not in (select mgr from emp where mgr is not null);  

    1. 一般不在子查询中使用order by

一般情况下,子查询使用order by或是不使用order by对主查询来说没有什么意义。子查询的结果给主查询当成集合来使用,所以没有必要将子查询order by。

但,在Top-N分析问题中,必须使用order by

    1. 一般先执行子查询,再执行主查询

含有子查询的SQL语句执行的顺序是,先子后主。

相关子查询例外

  1. 集合运算

查询部门号是10和20的员工信息: ?思考有几种方式解决该问题 ?

  1. SQL>  select * from emp where deptno in(10, 20)    
  2. SQL>  select * from emp where deptno=10 or deptno=20   
  3. 集合运算:

Select * from emp where deptno=10 加上

Select * from emp where deptno=20

集合运算所操作的对象是两个或者多个集合,而不再是表中的列(select一直在操作表中的列)

集合运算符

集合运算的操作符。A∩B、A∪ B、A - B

QL>   select * from emp where deptno=10

union

    select * from emp where deptno=20 注意这是一条SQL语句。

集合运算需要注意的问题

  1. 参与运算的各个集合必须列数相同,且类型一致
  2. 采用第一个集合的表头作为最终使用的表头。 (别名也只能在第一个集合上起)
  3. 可以使用括号()先执行后面的语句。

问题:按照部门统计各部门不同工种的工资情况,要求按如下格式输出:

 

分析SQL执行结果。  

第一部分数据是按照deptno和job进行分组;select 查询deptno、job、sum(sal)

第二部分数据是直接按照deptno分组即可,与job无关;select 只需要查询deptno,sum(sal)

第三部分数据不按照任何条件分组,即group by null;select 查询sum(sal)

所以,整体查询结果应该=  group by deptno,job  +  group by deptno  +  group by null

按照集合的要求,必须列数相同,类型一致,所以写法如下,使用null强行占位!

SQL> select deptno,job,sum(sal) from emp group by deptno,job

 union

 select deptno,to_char(null),sum(sal) from emp group by deptno

 union

 select to_number(null),to_char(null),sum(sal) from emp;

需要注意:集合运算的性能一般较差.

SQL的执行时间:

set timing on/off 默认是off

交集和差集与并集类似,也要注意以上三点。只不过算法不同而已。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/583260.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

YHZ005 Window 下安装 IDLE 、IPython 以及PyCharm 等开发工具

目录 &#x1f998; Window 下&#x1f408; IDLE&#x1f9a2; IPython&#x1f432; PyCharm 资源编号&#xff1a;YHZ005 配套视频&#xff1a;https://www.bilibili.com/video/BV1zy4y1Z7nk?p6 &#x1f998; Window 下 &#x1f408; IDLE 具体请见视频操作&#xff1a;…

深入理解 YOLOv8:解析.yaml 配置文件目标检测、实例分割、图像分类、姿态检测

目录 yolov8导航 YOLOv8&#xff08;附带各种任务详细说明链接&#xff09; 引言 YOLOv8配置文件概览 yolov8.yaml 1. nc 2. scales 3. backbone 4. head yolov8-seg.yaml 1. 参数部分 2. 骨架&#xff08;Backbone&#xff09;部分 3. 头部&#xff08;Head&…

《MySQL》事务篇

事务特性 ACID Atomicity原子性&#xff1a;事务中的操作要么全部完成&#xff0c;要么全部失败。 Consistency一致性&#xff1a;事务操作前后&#xff0c;数据满足完整性约束。 Isolation隔离性&#xff1a;允许并发执行事务&#xff0c;每个事务都有自己的数据空间&…

项目 引入 uView

安装 npm install uview-ui //或 yarn add uview-ui main.js引入 import Vue from vue; import uView from uview-ui;Vue.use(uView);//或// main.js import uView from /node_modules/uview-ui Vue.use(uView) uni.scss引入 import /node_modules/uview-ui/theme.scss…

2024年PMP考试新考纲-PMBOK第七版-【裁剪】真题解析

距离2024年3月份PMP考试的脚步越来越近了&#xff0c;如何快速、有效的备考3月份PMP考试呢&#xff1f;华研荟结合多年的PMP培训和辅导经验&#xff0c;前面的文章为大家提出了三个建议&#xff0c;只要按照这三个步骤走&#xff0c;现在从零开始也是完全有可能3A取得PMP证书的…

数据库是否可以直接作为数据仓库的数据源

在数据仓库使用数据时&#xff0c;我们是否可以直接将数据库作为数据源&#xff1f;如果使用了&#xff0c;会存在哪些问题&#xff1f; 数据库中存储的是业务数据&#xff0c;存储方式是行式存储&#xff1b;而数据仓库中数据是以列式存储的&#xff1b;如果数据仓库要想使用…

Android画布Canvas矩阵Matrix放大裁剪Rect区域的Bitmap,Kotlin

Android画布Canvas矩阵Matrix放大裁剪Rect区域的Bitmap&#xff0c;Kotlin private fun mydraw() {val originBmp BitmapFactory.decodeResource(resources, R.mipmap.pic).copy(Bitmap.Config.ARGB_8888, true)val newBmp Bitmap.createBitmap(originBmp.width, originBmp.h…

AI大模型

目录 前言 AGI通用人工智能 总结 前言 AI零基础直播公开课&#xff0c;了解一下。 AGI通用人工智能 query chatmind.tech 找一下这篇论文&#xff0c;了解一下 E为编码器&#xff0c;D为解码器 所谓大模型也就是编码器-解码器。 模型里会有一些公式和参数&#xff0c;厉害的…

go 源码解读 sync.RWMutex

sync.RWMutex 简介源码结构RLockRUnlockUnlockgo 运行时方法 简介 简述sync包中读写锁的源码。 &#xff08;go -version 1.21&#xff09; 读写锁&#xff08;RWMutex&#xff09;是一种并发控制机制&#xff0c;用于在多个 goroutine 之间对共享资源进行读写操作。它提供了…

浅谈WPF之控件模板Control Template和数据模板Data Template

WPF不仅支持传统的Windows Forms编程的用户界面和用户体验设计&#xff0c;同时还推出了以模板为核心的新一代设计理念。在WPF中&#xff0c;通过引入模板&#xff0c;将数据和算法的“内容”和“形式”进行解耦。模板主要分为两大类&#xff1a;数据模板【Data Template】和控…

C语言——数据类型

一、基本类型&#xff1a; 它们是构建其他数据类型的基础。 1、整型 用于表示整数。例如&#xff0c;int a 10; 整形中又有许多类型&#xff1a; 类型大小范围int2 或 4 字节-32,768 到 32,767 或 -2,147,483,648 到 2,147,483,647unsigned int2 或 4 字节0 到 65,535 或…

【zookeeper分布式锁】

文章目录 1.Zookeeper 分布式锁实战 1.Zookeeper 分布式锁实战 1.1 什么是分布式锁 在单体的应用开发场景中涉及并发同步的时候&#xff0c;大家往往采用Synchronized&#xff08;同步&#xff09;或者其他同一个 JVM内Lock机制来解决多线程间的同步问题。在分布式集群工作的开…

《MySQL系列-InnoDB引擎01》MySQL体系结构和存储引擎

文章目录 第一章 MySQL体系结构和存储引擎1 数据库和实例2 MySQL配置文件3 MySQL数据库路径4 MySQL体系结构5 MySQL存储引擎5.1 InnoDB存储引擎5.2 MyISAM存储引擎5.3 NDB存储引擎5.4 Memory存储引擎5.5 Archive存储引擎5.6 Federated存储引擎 6 连接MySQL6.1 TCP/IP6.2 命名管…

关于“Python”的核心知识点整理大全48

目录 world_population.py 16.2.5 制作世界地图 americas.py 16.2.6 在世界地图上呈现数字数据 na_populations.py 16.2.7 绘制完整的世界人口地图 world_population.py 16.2.8 根据人口数量将国家分组 world_population.py 16.2.9 使用 Pygal 设置世界地图的样式 w…

ubuntu快速搭建java开发环境/java1.8/idea2021.1.3/mysql/doceker

当设置Java开发环境时&#xff0c;确保先安装Java Development Kit (JDK) 8、IntelliJ IDEA 2021.1.3专业版、Maven、Git、MySQL和Docker。以下是Ubuntu上安装这些工具的基本步骤&#xff1a; 1. 安装Java Development Kit (JDK) 8&#xff1a; 首先&#xff0c;更新APT软件包…

Presentation Error:编程中的细节之战

Presentation Error&#xff1a;编程中的细节之战 大家好&#xff0c;我是免费搭建查券返利机器人赚佣金就用微赚淘客系统3.0的小编&#xff0c;也是冬天不穿秋裤&#xff0c;天冷也要风度的程序猿&#xff01;今天&#xff0c;让我们一起探讨在程序设计和编程中常见的问题之一…

PECL 到 LVPECL 的接口使用交流耦合或 3 电阻端接。采用交流耦合作热拔插时需注意防止因电容积累电荷放电导致器件损伤

交流耦合的必要性 PECL和LVPECL信号是差分信号,设计成具有一定的直流偏置电压。这种直流偏置电压在不同的设备或电路板之间可能会有所不同,直接连接可能导致直流偏置电平的冲突,从而损坏器件或影响信号完整性。交流耦合通过串联电容来隔离直流成分,只允许交流信号通过,从而…

web安全,常见的攻击以及如何防御

1、CSRF攻击 CSRF即Cross-site request forgery(跨站请求伪造) &#xff08;1&#xff09;为了防止这种攻击&#xff0c;表单一般都带有一个随机 token&#xff0c;告诉服务器这是真实请求。 <form action"your-bank.com/transfer" method"POST">&…

docker小白第九天

docker小白第九天 安装redis集群 cluster(集群)模式-docker版本&#xff0c;哈希槽分区进行亿级数据存储。如果1~2亿条数据需要缓存&#xff0c;请问如何设计这个存储案例。单机存储是不可能的&#xff0c;需要分布式存储&#xff0c;如果使用redis又该如何部署。 哈希取余分…

5-Docker实例-centos-nginx(3)

基于上次制作的centos-systemctl:1.0镜像,在其上安装nginx rpm,并构建nginx镜像。 1.制作nginx Dockerfile,并保存 命令: vim Dockerfile FROM centos-systemctl:1.0 MAINTAINER ztjCOPY nginx.repo /etc/yum.repos.d/ COPY epel.repo /etc/yum.repos.d/ COPY epel-tes…