【Oracle】Oracle系列之八--SQL查询

news2025/1/15 19:39:56

文章目录

  • 往期回顾
  • 前言
  • 1. 基本查询
    • (1)All
    • (2)in/exists 子查询
    • (3)union/except/intersect
    • (4)group by
    • (5)having
    • (6)聚集函数:avg, max, min, sum, count
  • 2. 连接查询
    • (1)内连接
    • (2)外连接
    • (3)交叉连接交叉连接不带WHERE子句,它返回被连接的两个表所有数据行的笛卡尔积,返回到结果集合中的数据行数等于第一个表中符合查询条件的数据行数乘以第二个表中符合查询条件的数据行数。
  • 3. SQL语句解析
    • (1)分析
    • (2)优化
    • (3)执行
  • 4. Oracle Hint
    • (1)优化器模式提示
    • (2)访问路径提示
    • (3)并行执行提示
    • (4)其他

往期回顾

  • 【Oracle】Oracle系列–Oracle数据类型
  • 【Oracle】Oracle系列之二–Oracle数据字典
  • 【Oracle】Oracle系列之三–Oracle字符集
  • 【Oracle】Oracle系列之四–用户管理
  • 【Oracle】Oracle系列之五–Oracle表空间
  • 【Oracle】Oracle系列之六–Oracle表分区
  • 【Oracle】Oracle系列之七–表的创建与管理

前言

Oracle提供了一个强大的SQL引擎,使得用户可以通过SQL语言来管理和操作数据库。

1. 基本查询

以CAP(顾客-代理-产品)数据库为例,表结构如下:

  • CUSTOMERS(顾客信息表)
    • cid 顾客ID
    • cname 顾客姓名
    • city 顾客所在城市
    • discnt 顾客可能会有的折扣
  • AGENTS 代理商信息表
    • aid 代理商ID
    • aname 代理商名称
    • city 代理商所在城市
    • percent 代理商每笔交易所能获得佣金的百分比
  • PRODUCTS( 商品信息表 )
    • pid 商品ID
    • pname 商品名称
    • city 商品库存所在城市
    • quantity 商品库存数量
    • price 商品批发价
  • ORDERS( 订单信息表)
  • ordno 订单ID
  • Month 订单月份
  • cid 顾客ID
  • aid 代理商ID
  • pid 商品ID
  • qty 数量
  • dollars 商品总价

(1)All

找出佣金百分率最小的代理商aid

select aid from agents where percent <=all (select percent from agents);

(2)in/exists 子查询

EXISTS用于检查subquery是否至少会返回一行数据,subquery 是一个受限的 SELECT 语句 (不允许有 COMPUTE 子句和 INTO 关键字) 返回一个结果集,EXISTS子句根据其内查询语句的结果集空或者非空,返回一个布尔值True或False。一种通俗的可以理解为:将外查询表的每一行,代入内查询作为检验,如果内查询返回的结果取非空值,则EXISTS子句返回TRUE,这一行行可作为外查询的结果行,否则不能作为结果。
NOT EXISTS 的作用与 EXISTS 正好相反。如果子查询没有返回行,则满足了 NOT EXISTS 中的 WHERE 子句。

  • 找出既订购了产品p01有订购了产品p07的顾客cid
    关系代数

SQL语句

select distinct cid from orders x where x.pid=‘p01’ and exists (select * from orders y where x.cid=y.cid and y.pid=‘p07’);

select distinct x.cid from orders x,orders y where x.pid=‘p01’ and x.cid=y.cid and y.cid=‘p07’ ;

找出没有通过代理商a03订货的顾客cid
关系代数:
在这里插入图片描述

SQL语句

select distinct cid from orders x where not exists (select * from orders where cid=x.cid and  aid=‘a03’);

EXISTS的查询一般能找到等价的其他查询形式,如

select distinct t.xk from bm_zyml t where not exists (select * from bm_xk s where s.mc= t.xk);

等价于

select distinct t.xk from bm_zyml t where t.xk not in (select s.mc from bm_xk s)

通常情况下采用exists要比in效率高,因为in不走索引,但一般in适合于外表大而内表小的情况,exists适合于外表小而内表大的情况。

(3)union/except/intersect

union
包含了顾客所在的或代理商所在或两者皆在的城市名单:

select city from customers union select city from agents;#不含重复行
select city from customers union all select city from agents;# 含有重复行

except:返回在 TABLE1 中但不在 TABLE2 中的行并消除所有重复行。
intersect:返回 TABLE1 和 TABLE2 中都有的行并消除所有重复行。

(4)group by

每个代理商为顾客c002和c003订购产品及产品总数量:

select a.aid,aname,p.pid,pname,sum(qty) 
from orders x,products p,agents a
where x.pid=p.pid and x.aid=a.aid 
  and x.cid in (‘c002’,’c003’)
group by a.aid,a.aname,p.pid,p.pname;

group by 后可跟多个字段

(5)having

至少两个顾客订购的产品pid:

select pid from orders 
group by pid having count(distinct cid) >=2

(6)聚集函数:avg, max, min, sum, count

聚焦函数不能作为条件用在where子句中,需要与having,group一起使用

所有代理商的最大销售额的平均值:

select avg(select max(dollars) from orders group by  aid);

删除总订货金额小于600的代理商:

 delete from agents where aid in (select aid from orders group by aid having sum(dollars)<600); 

2. 连接查询

例如图书馆借阅系统,表book与student结构如下:

(1)内连接

内连接查询操作列出与连接条件匹配的数据行,它使用比较运算符比较被连接列的列值。
上述SQL等价于

select * from book t, student s where t.studentid=s.studentid

(2)外连接

外连接返回到查询结果集合中的不仅包含符合连接条件的行,而且还包括左表(左外连接时)、右表(右外连接时)或两个边接表(全外连接)中的所有数据行,无匹配的显示空值。

左外连接(left join)

上述SQL等价于

select * from book t, student s where t.studentid(+) = s.studentid

右外连接(right join)

上述SQL等价于

select * from book t, student s where t.studentid = s.studentid(+)

全外连接(full join)

(3)交叉连接交叉连接不带WHERE子句,它返回被连接的两个表所有数据行的笛卡尔积,返回到结果集合中的数据行数等于第一个表中符合查询条件的数据行数乘以第二个表中符合查询条件的数据行数。

3. SQL语句解析

SQL解析指的是Oracle数据库将用户提交的SQL语句转换成可以执行的代码的过程,主要包括三个阶段:分析、优化和执行。

(1)分析

符号检查和词法分析:Oracle会检查SQL语句中的每个字符,确保都是有效的标识符,并且符合数据库中定义的规则。如果Oracle发现了任何错误或不合法的字符,它将抛出一个错误,并停止解析过程。
语法分析:Oracle根据SQL语句的语法规则,将其分解为多个操作和表达式。例如,SQL语句中的SELECT语句可以被分解为列名、FROM子句、WHERE子句等部分。如果Oracle发现了任何语法错误,它也会抛出一个错误并停止解析过程。
语义分析:Oracle将确定SQL语句中每个表、列名和函数是否存在、是否被正确引用,并且它们的数据类型是否兼容。如果Oracle发现了任何不兼容的类型或其他不一致的引用,它也会抛出一个错误并停止解析过程。

(2)优化

优化器处理:Oracle将执行一系列的优化规则,以便找到SQL语句的最佳执行计划。这些规则包括索引选择、连接顺序和扫描类型等。由于优化器处理需要大量的计算和IO操作,所以在这个步骤中可能会花费大量的时间。

(3)执行

执行计划生成:Oracle将使用优化器处理过的最佳执行计划,生成一个具体的执行计划。执行计划通常是一个树形结构,它描述了SQL语句将使用哪些操作、连接和过滤器,以及它们的执行顺序。
执行:Oracle将执行执行计划,并返回查询结果。根据SQL语句的类型,可能会涉及到许多不同的操作,包括表扫描、索引扫描、连接和聚合等。如果有任何错误,它们将在执行过程中被捕获,并且数据库将返回错误代码。
SQL语句解析的过程是非常复杂和计算密集的,每个步骤都需要大量的处理和IO操作。因此,如果SQL语句不正确或不规范,它会导致解析速度缓慢甚至失败。另外,一个错误的执行计划也会导致查询效率低下或查询结果不正确。为了确保最佳的数据库性能,必须深入了解SQL语句解析的过程,并可以优化查询语句以提高查询效率。

e.g. 标准的SQL 解析顺序:

(1)FROM 子句, 组装来自不同数据源的数据
(2)WHERE 子句, 基于指定的条件对记录进行筛选
(3)GROUP BY 子句, 将数据划分为多个分组
(4)使用聚合函数进行计算
(5)使用 HAVING 子句筛选分组
(6)计算所有的表达式
(7)使用 ORDER BY 对结果集进行排序
例如:在学生成绩表中 (暂记为 tb_Grade), 把 "考生姓名"内容不为空的记录按照 “考生姓名” 分组, 并且筛选分组结果, 选出 “总成绩” 大于 600 分的。

SQL 语句为:

select 考生姓名, max(总成绩) as max总成绩 
from tb_Grade 
where 考生姓名 is not null 
group by 考生姓名 
having max(总成绩) > 600 
order by max总成绩 

执行顺序如下:

(1) 首先执行 FROM 子句, 从 tb_Grade 表组装数据源的数据
(2) 执行 WHERE 子句, 筛选 tb_Grade 表中所有数据不为 NULL 的数据
(3) 执行 GROUP BY 子句, 把 tb_Grade 表按 “学生姓名” 列进行分组
(4) 计算 max() 聚集函数, 按 “总成绩” 求出总成绩中最大的一些数值
(5) 执行 HAVING 子句, 筛选课程的总成绩大于 600 分的.
(6) 执行 ORDER BY 子句, 把最后的结果按 “Max 成绩” 进行排序

4. Oracle Hint

Oracle hint是一种在SQL语句中使用的特殊注释,它可以告诉Oracle数据库如何执行SQL语句,从而达到最优的执行效果。

Oracle hint主要用于优化复杂的SQL查询语句,特别是当优化器无法选择最优的执行计划时。例如,当使用复杂的连接查询、子查询、聚合函数或大数据量的表时,Oracle hint通过给出提示指导优化器选择最优的执行计划,从而提高SQL语句的执行效率和性能。

Oracle hint必须写在SQL语句的SELECT、INSERT、UPDATE或DELETE语句之后,但在WHERE子句之前。

注意:如果表中指定了别名,那么Hint中也必须使用别名,否则Hint会忽略:
Select /+full(a)/ * from t a; – 使用hint
Select /*+full(t) */ * from t a; --不使用hint

(1)优化器模式提示

对语句块选择基于开销的优化方法,并获得最佳吞吐量,使资源消耗最小化。

select /*+ALL_ROWS(t)*/* from student t Where name='WASEEM HAIDER'
select /*+FIRST_ROWS(t)*/* from student t Where name ='WASEEM HAIDER'
select /*+FIRST_ROWS(t,20)*/* from student t Where name ='WASEEM HAIDER'

ALL_ROWS时,Oracle 会用最快的速度将SQL执行完毕,将所有结果集全部返回,在OLAP 系统中使用得比较多;ALL_ROWS 强调SQL整体的执行效率,而FIRST_ROWS(n)强调用最快的速度返回前N行,而不管所有的结果返回的时长,可能最后一条要很长时间才能获得。

对语句块选择基于规则的优化方法

select /*+RULE(t)*/* from student t Where name ='WASEEM HAIDER'

(2)访问路径提示

对表选择全局扫描的方法

select /*+FULL(t)*/* from student t Where name='WASEEM HAIDER'

对表选择索引的扫描方法

select /*+INDEX(T_JBXX X)*/* from student  Where name='WASEEM HAIDER'

(3)并行执行提示

select /*+PARALLEL(t,16)*/* from student t  Where t.name ='WASEEM HAIDER'

这个值会覆盖表自身设定的并行度,如果这个值为default,CBO使用系统参数值。

(4)其他

Insert /*+append */ into t as select * from all_objects

提示数据库以直接加载的方式(direct load)将数据加载入库,尤其在插入大量的数据,一般都会用此hint。

此外还有表连接顺序/表关联方式提示、查询转换提示等。

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

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

相关文章

SLAM从入门到精通(用c++实现机器人运动控制)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 之前的一篇文章&#xff0c;我们知道了可以通过wpr_simulation包仿真出机器人和现场环境。如果需要控制机器人&#xff0c;这个时候就需要rqt_robo…

AcWing 5153. 删除(AcWing杯 - 周赛)(结论+枚举)

思路&#xff1a; ACcode: #include<bits/stdc.h> using namespace std; #define int long long string s; void solve() {cin>>s;s"00"s;int lens.size();for(int i0; i<len; i) {for(int ji1; j<len; j) {for(int kj1; k<len; k) {int xs[i]*…

leetcode:2446. 判断两个事件是否存在冲突(python3解法)

难度&#xff1a;简单 给你两个字符串数组 event1 和 event2 &#xff0c;表示发生在同一天的两个闭区间时间段事件&#xff0c;其中&#xff1a; event1 [startTime1, endTime1] 且event2 [startTime2, endTime2] 事件的时间为有效的 24 小时制且按 HH:MM 格式给出。 当两个…

Windows 基于Visual Studio 开发Qt 6 连接MySQL 8

前提条件&#xff1a; 1、Visual Studio 2022 社区版(免费版) 2、Qt-6.5.1版本 3、MySQL 8 Qt 6 配置MySQL 8 动态/静态连接库和MySQL 8 驱动。 libmysql.dll 和libmysql.lib是QT所需的动态和静态链接库&#xff1b;qsqlmysql.dll 和qsqlmysql.dll.debug是Qt所需的mysql驱…

机器人过程自动化(RPA)入门 1. 什么是机器人过程自动化?

如今&#xff0c;我们生活中几乎没有任何方面不受自动化的影响。一些例子包括洗衣机、微波炉、汽车和飞机的自动驾驶模式&#xff0c;雀巢在日本的商店里使用机器人销售咖啡豆&#xff0c;沃尔玛在美国测试无人机送货&#xff0c;我们的银行支票使用光学字符识别&#xff08;OC…

【Linux】调试代码的工具 - gdb

1、安装gdb sudo yum -y install gdb【安装gdb】 2、gdb的使用 在 Linux 下&#xff0c;我们编写代码默认以 release 方式发布如果想让代码以 debug 方式发布&#xff0c;必须给 gcc 添加 -g 选项 (gdb) q / quit【退出gdb】(gdb) l / list&#xff08;list可缩写为 l&#xf…

C++的文件操作

文件操作 程序运行时产生的数据都属于临时数据&#xff0c;通过文件可将数据持久化 C中对文件操作需要包含头文件<fstream> 文件类型分为两种&#xff1a; 文本文件 - 文件以文本的ASCII码形式存储在计算机中二进制文件 - 文件以文本的二进制形式存储在计算机中&…

如何取消自动播放音乐:取消手机汽车连上后汽车自动播放音乐?

背景 手机和汽车通过蓝牙连接上之后&#xff0c;汽车音响会自动播放手机上的音乐&#xff0c;似乎是自动唤醒APP的&#xff0c;因为这些音乐APP在手机上是已经被杀了后台的了。 而且汽车的屏幕的播放列表里头会显示播放的音乐的名称&#xff0c;也有可能是视频的名称&#xf…

安卓备份分区----手动查询安卓系统分区信息 导出系统分区的一些基本操作

在玩机搞机过程中。有时候需要手动查看有些分区信息&#xff0c;或者备份分区的操作。那么今天以小米8为例解析下其中的操作步骤 机型&#xff1a;小米8 adb版本&#xff1a;https://developer.android.com/studio/releases/platform-tools 机型芯片&#xff1a;高通骁龙845…

基于微信小程序的校园商铺系统,附源码、数据库

文章目录 第一章 简介第二章 技术栈第三章&#xff1a;总体设计第四章系统详细设计4.1 前台功能模块4.2后台功能模块4.2.1管理员功能模块 五 源码咨询 第一章 简介 今天&#xff0c;为大家带来的事基于微信小程序的校园商铺系统。本系统的主要意义在于&#xff0c;全力以赴为用…

Redis双写一致性、持久化机制、分布式锁

一.双写一致性: 含义:当数据库中的数据被修改了以后&#xff0c;我们也需要同时修改缓存&#xff0c;使缓存和数据库的数据保持一致 &#xff08;1&#xff09;读操作:当请求发来的时候&#xff0c;先去看redis里面是否有对应的数据&#xff0c;如果有直接返回&#xff0c;如果…

轻量级的日志采集组件 Filebeat 讲解与实战操作

文章目录 一、概述二、Kafka 安装三、Filebeat 安装1&#xff09;下载 Filebeat2&#xff09;Filebeat 配置参数讲解3&#xff09;filebeat.prospectors 推送kafka完整配置1、filebeat.prospectors2、processors3、output.kafka 4&#xff09;filebeat.inputs 与 filebeat.pros…

【STL】vector常见用法及模拟实现(附源码)

目录 前言1. vector介绍及使用1.1vector的介绍1.2 vector的使用1.2.1 构造函数 1.2.2 vector对象遍历1.2.3 reserve和resize1.2.4 insert和erase 2. vector模拟实现2.1 vector迭代器失效问题2.2 模拟实现reserve函数浅拷贝问题2.3模拟实现源码2.3.1 vector.h2.3.2 test.cpp 前言…

org.postgresql.util.PSQLException: Bad value for type long

项目用 springbootmybatis mybatisplus&#xff0c; 数据库是&#xff1a;postgresql 。 执行查询时候返回错误。 org.springframework.dao.DataIntegrityViolationException: Error attempting to get column city_id from result set. Cause: org.postgresql.util.PSQLExce…

如何让ChatGPT为留学生所用?

“我们这一届学Data Analyics和Data Science的没一个找到工作的。”朋友饭桌上的闲话让研究生才算踏入DA圈子的我瑟瑟发抖。 还没开始正式求职的我&#xff0c;似乎已经被宣告失业了。而这一切都要“归功”于以ChatGPT为代表的大语言模型&#xff08;LLMs&#xff09;。 问世不…

接口测试练习步骤

在接触接口测试过程中补了很多课&#xff0c; 终于有点领悟接口测试的根本&#xff1b; 偶是个实用派&#xff5e;&#xff0c;那么现实中没有用的东西&#xff0c;基本上我都不会有很大的概念&#xff1b; 下面给的是接口测试的统一大步骤&#xff0c;其实就是让我们对接口…

第9章 【MySQL】InnoDB的表空间

表空间 是一个抽象的概念&#xff0c;对于系统表空间来说&#xff0c;对应着文件系统中一个或多个实际文件&#xff1b;对于每个独立表空间来说&#xff0c;对应着文件系统中一个名为 表名.ibd 的实际文件。大家可以把表空间想象成被切分为许许多多个 页 的池子&#xff0c;当我…

机器学习之正则化与验证提高模型泛化

文章目录 正则化&#xff08;Regularization&#xff09;&#xff1a;验证&#xff08;Validation&#xff09;&#xff1a; 正则化和验证是机器学习中重要的概念&#xff0c;它们帮助提高模型的性能和泛化能力。让我详细介绍一下这两个概念&#xff1a; 正则化&#xff08;Re…

【Git】轻松学会 Git:深入理解 Git 的基本操作

文章目录 前言一、创建 Git 本地仓库1.1 什么是仓库1.2 创建本地仓库1.3 .git 目录结构 二、配置 Git三、认识 Git 的工作区、暂存区和版本库3.1 什么是 Git 的工作区、暂存区和版本库3.2 工作区、暂存区和版本库之间的关系 四、添加文件4.1 添加文件到暂存区和版本库中的命令4…

php文件上传功能(文件上传)

实现文件上传是Web开发中常用的功能之一&#xff0c;而PHP也是支持文件上传的。那么&#xff0c;下面我们就来介绍一下常用的PHP实现文件上传的方法。 使用HTML表单实现文件上传 HTML表单是Web开发中最基本的元素之一&#xff0c;它可以接收用户输入的数据&#xff0c;并通过…