数据库系统概论(超详解!!!) 第四节 关系数据库标准语言SQL(Ⅲ)

本文主要是介绍数据库系统概论(超详解!!!) 第四节 关系数据库标准语言SQL(Ⅲ),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.连接查询

连接查询:同时涉及多个表的查询

连接条件或连接谓词:用来连接两个表的条件     

一般格式:

[<表名1>.]<列名1>  <比较运算符>  [<表名2>.]<列名2>

[<表名1>.]<列名1> BETWEEN [<表名2>.]<列名2> AND [<表名2>.]<列名3>

连接字段:连接谓词中的列名称

连接条件中的各连接字段类型必须是可比的,但名字不必是相同的

查询每个学生及其选修课程的情况SELECT  Student.*,SC.*  FROM  Student,SC WHERE  Student.Sno = SC.Sno;

1、等值与非等值连接查询

等值连接:连接运算符为=

查询每个学生及其选修课程的情况SELECT  Student.*,SC.*FROM     Student,SCWHERE  Student.Sno = SC.Sno;

连接操作的执行过程:

嵌套循环法(NESTED-LOOP)

首先在表1中找到第一个元组,然后从头开始扫描表2,逐一查找满足连接件的元组,找到后就将表1中的第一个元组与该元组拼接起来,形成结果表中一个元组。

表2全部查找完后,再找表1中第二个元组,然后再从头开始扫描表2,逐一查找满足连接条件的元组,找到后就将表1中的第二个元组与该元组拼接起来,形成结果表中一个元组。

重复上述操作,直到表1中的全部元组都处理完毕

排序合并法(SORT-MERGE)

常用于=连接

首先按连接属性对表1和表2排序

对表1的第一个元组,从头开始扫描表2,顺序查找满足连接条件的元组,找到后就将表1中的第一个元组与该元组拼接起来,形成结果表中一个元组。当遇到表2中第一条大于表1连接字段值的元组时,对表2的查询不再继续

找到表1的第二条元组,然后从刚才的中断点处继续顺序扫描表2,查找满足连接条件的元组,找到后就将表1中的第二个元组与该元组拼接起来,形成结果表中一个元组。直接遇到表2中大于表1连接字段值的元组时,对表2的查询不再继续

重复上述操作,直到表1或表2中的全部元组都处理完毕为止

索引连接(INDEX-JOIN)

对表2按连接字段建立索引

对表1中的每个元组,依次根据其连接字段值查询表2的索引,从中找到满足条件的元组,找到后就将表1中的第一个元组与该元组拼接起来,形成结果表中一个元组 


查询每个学生及其选修课程的情况,用自然连接完成。SELECT  Student.Sno,Sname,Ssex,Sage,Sdept,Cno,GradeFROM     Student,SCWHERE  Student.Sno = SC.Sno;

一条SQL语句可以同时完成选择和连接查询,这时WHERE子句是由连接谓词和选择谓词组成的复合条件。

查询选修2号课程且成绩在90分以上的所有学生的学号和姓名。SELECT Student.Sno, SnameFROM     Student, SCWHERE  Student.Sno=SC.Sno  AND    		               SC.Cno='2' AND SC.Grade>90;

执行过程:

先从SC中挑选出Cno='2'并且Grade>90的元组形成一个中间关系

再和Student中满足连接条件的元组进行连接得到最终的结果关系

2、自身连接

自身连接:一个表与其自己进行连接

需要给表起别名以示区别

由于所有属性名都是同名属性,因此必须使用别名前缀

查询每一门课的间接先修课(即先修课的先修课)SELECT  FIRST.Cno,SECOND.CpnoFROM  Course  FIRST,Course  SECONDWHERE FIRST.Cpno = SECOND.Cno;

3、外连接

外连接与普通连接的区别:

普通连接操作只输出满足连接条件的元组

外连接操作以指定表为连接主体,将主体表中不满足连接条件的元组一并输出  

左外连接: 列出左边关系中所有的元组  

右外连接: 列出右边关系中所有的元组

全外连接

查询每个学生及其选修课程的情况
SELECT Student.Sno,Sname,Ssex,Sage,Sdept,Cno,GradeFROM  Student  LEFT OUTER JOIN SC ON    (Student.Sno=SC.Sno); 

4、复合条件连接(多表连接)

多表连接:两个以上的表进行连接

查询每个学生的学号、姓名、选修的课程名及成绩SELECT Student.Sno, Sname, Cname, GradeFROM    Student, SC, Course    /*多表连接*/WHERE Student.Sno = SC.Sno AND SC.Cno = Course.Cno; /*符合条件*/

2.嵌套查询

嵌套查询概述

一个SELECT-FROM-WHERE语句称为一个查询块

将一个查询块嵌套在另一个查询块的WHERE子句或HAVING短语的条件中的查询称为嵌套查询

SELECT Sname		           /*外层查询/父查询*/FROM StudentWHERE Sno IN(SELECT Sno             /*内层查询/子查询*/FROM SCWHERE Cno= ' 2 ');

上层的查询块称为外层查询或父查询

下层查询块称为内层查询或子查询

SQL语言允许多层嵌套查询 :即一个子查询中还可以嵌套其他子查询

子查询的限制 :不能使用ORDER BY子句

不相关子查询:    

子查询的查询条件不依赖于父查询, 由里向外逐层处理。即每个子查询在上一级查询处理之前求解,子查询的结果用于建立其父查询的查找条件。

相关子查询:

子查询的查询条件依赖于父查询 。首先取外层查询中表的第一个元组,根据它与内层查询相关的属性值处理内层查询,若WHERE子句返回值为真,则取此元组放入结果表。 然后再取外层表的下一个元组 重复这一过程,直至外层表全部检查完为止。

1、带有IN谓词的子查询  

查询与“刘晨”在同一个系学习的学生。
此查询要求可以分步来完成① 确定“刘晨”所在系名             SELECT  Sdept  FROM     Student                            WHERE  Sname= ' 刘晨 ';结果为: CS
② 查找所有在CS系学习的学生。    SELECT   Sno,Sname,Sdept     FROM      Student                 WHERE  Sdept= ' CS '; 查询与“刘晨”在同一个系学习的学生。
将第一步查询嵌入到第二步查询的条件中SELECT Sno,Sname,SdeptFROM StudentWHERE Sdept  IN(SELECT SdeptFROM StudentWHERE Sname= ‘ 刘晨 ’);此查询为不相关子查询。用自身连接完成查询要求SELECT  S1.Sno,S1.Sname,S1.SdeptFROM     Student S1,Student S2WHERE  S1.Sdept = S2.Sdept  ANDS2.Sname = '刘晨';查询选修了课程名为“信息系统”的学生学号和姓名SELECT Sno,Sname                       ③ 最后在Student关系中FROM    Student                                     取出Sno和SnameWHERE Sno  IN(SELECT Sno                              ② 然后在SC关系中找出选FROM    SC                                    修了3号课程的学生学号WHERE  Cno IN(SELECT Cno                       ① 首先在Course关系中找出FROM Course                      “信息系统”的课程号,为3号WHERE Cname= ‘信息系统’));用连接查询实现SELECT Sno,SnameFROM    Student,SC,CourseWHERE Student.Sno = SC.Sno  ANDSC.Cno = Course.Cno ANDCourse.Cname=‘信息系统’;

2、 带有比较运算符的子查询  

当能确切知道内层查询返回单值时,可用比较运算符(>,<,=,>=,<=,!=或< >)。

与ANY或ALL谓词配合使用

假设一个学生只可能在一个系学习,并且必须属于一个系,则在
查询每个学生的学号、姓名、选修的课程名及成绩
可以用 = 代替IN :SELECT Sno,Sname,SdeptFROM    StudentWHERE Sdept   =(SELECT SdeptFROM    StudentWHERE Sname= ‘刘晨’);

子查询一定要跟在比较符之后

找出每个学生超过他选修课程平均成绩的课程号。SELECT Sno, CnoFROM  SC  xWHERE Grade >=(SELECT AVG(Grade) FROM  SC yWHERE y.Sno=x.Sno);从外层查询中取出SC的一个元组x,将元组x的Sno值(200215121)传送给内层查询。SELECT AVG(Grade)FROM SC yWHERE y.Sno='200215121';执行内层查询,得到值88(近似值),用该值代替内层查询,得到外层查询:SELECT Sno, CnoFROM  SC xWHERE Grade >=88; 

3、 带有ANY(SOME)或ALL谓词的子查询  

使用ANY或ALL谓词时必须同时使用比较运算 ,语义为:      

> ANY    大于子查询结果中的某个值      

> ALL    大于子查询结果中的所有值

< ANY    小于子查询结果中的某个值    

< ALL    小于子查询结果中的所有值

>= ANY    大于等于子查询结果中的某个值    

>= ALL    大于等于子查询结果中的所有值

<= ANY    小于等于子查询结果中的某个值    

<= ALL    小于等于子查询结果中的所有值

= ANY    等于子查询结果中的某个值        

=ALL    等于子查询结果中的所有值(通常没有实际意义)

!=(或<>)ANY    不等于子查询结果中的某个值

!=(或<>)ALL    不等于子查询结果中的任何一个值

查询非计算机科学系中比计算机科学系任意一个学生年龄小的学生姓名和年龄SELECT Sname,SageFROM    StudentWHERE Sage < ANY (SELECT  SageFROM    StudentWHERE Sdept= ' CS ')AND Sdept <> ‘CS ' ;           /*父查询块中的条件 */执行过程:(1)首先处理子查询,找出CS系中所有学生的年龄,构成一个集合(20,19)(2)处理父查询,找所有不是CS系且年龄小于 20 或 19的学生用聚集函数实现SELECT Sname,SageFROM   StudentWHERE Sage < (SELECT MAX(Sage)FROM StudentWHERE Sdept= 'CS ')AND Sdept <> ' CS ';查询非计算机科学系中比计算机科学系所有学生年龄都小的学生姓名及年龄。
方法一:用ALL谓词SELECT Sname,SageFROM StudentWHERE Sage < ALL(SELECT SageFROM StudentWHERE Sdept= ' CS ')AND Sdept <> ' CS ’;方法二:用聚集函数SELECT Sname,SageFROM StudentWHERE Sage < (SELECT MIN(Sage)FROM StudentWHERE Sdept= ' CS ')AND Sdept <>' CS ';

ANY(或SOME),ALL谓词与聚集函数、IN谓词的等价转换关系

ca4a71db40ad45ba9faa7351302e1cbd.png

4、 带有EXISTS谓词的子查询

EXISTS谓词 :存在量词

 带有EXISTS谓词的子查询不返回任何数据,只产生逻辑真值“true”或逻辑假值“false”。

若内层查询结果非空,则外层的WHERE子句返回真值

若内层查询结果为空,则外层的WHERE子句返回假值

由EXISTS引出的子查询,其目标列表达式通常都用 * ,因为带EXISTS的子查询只返回真值或假值,给出列名无实际意义。

NOT EXISTS谓词 :

若内层查询结果非空,则外层的WHERE子句返回假值

若内层查询结果为空,则外层的WHERE子句返回真值

查询所有选修了1号课程的学生姓名。思路分析:
本查询涉及Student和SC关系
在Student中依次取每个元组的Sno值,用此值去检查SC表
若SC中存在这样的元组,其Sno值等于此Student.Sno值,并且其Cno= ‘1’,则取此Student.Sname送入结果表SELECT SnameFROM StudentWHERE EXISTS(SELECT *FROM SCWHERE Sno=Student.Sno AND Cno= ' 1 ');查询没有选修1号课程的学生姓名。SELECT SnameFROM     StudentWHERE NOT EXISTS(SELECT *FROM SCWHERE Sno = Student.Sno AND Cno='1');查询与“刘晨”在同一个系学习的学生。可以用带EXISTS谓词的子查询替换:SELECT Sno,Sname,SdeptFROM Student S1WHERE EXISTS(SELECT *FROM Student S2WHERE S2.Sdept = S1.Sdept ANDS2.Sname = '刘晨');

不同形式的查询间的替换

一些带EXISTS或NOT EXISTS谓词的子查询不能被其他形式的子查询等价替换

所有带IN谓词、比较运算符、ANY和ALL谓词的子查询都能用带EXISTS谓词的子查询等价替换

用EXISTS/NOT EXISTS实现全称量词(难点)

SQL语言中没有全称量词 ,(For all) 可以把带有全称量词的谓词转换为等价的带有存在量词的谓词:

171112a40491442c88c53af35b43e3d0.png

查询选修了全部课程的学生姓名。(不存在没有选修的课程)SELECT SnameFROM StudentWHERE NOT EXISTS(SELECT *FROM CourseWHERE NOT EXISTS(SELECT *FROM SCWHERE Sno= Student.SnoAND Cno= Course.Cno));

分步理解,从最下边的where看起,从下往上分析。

1、最内层的select是把学生学号和课程号带入,结果是查询学生选课的记录,加上not exists,就是学生没选的课程。

2、再加上上层的select(从课程表),就是选出学生没选的课程。

3、最后在上面又加了一个not exists,就是不存在学生没选的课程(既学生选了所有的课程才会符合记录)

用EXISTS/NOT EXISTS实现逻辑蕴涵(难点)

SQL语言中没有蕴涵(Implication)逻辑运算 ,可以利用谓词演算将逻辑蕴涵谓词等价转换为:

9975c9b1e5e2423da7cfa4c6cdea94f6.png

查询至少选修了学生201215122选修的全部课程的学生号码。

解题思路:

用逻辑蕴涵表达:查询学号为x的学生,对所有的课程y,只要201215122学生选修了课程y,则x也选修了y。

形式化表示:     

用P表示谓词 “学生201215122选修了课程y”     

用q表示谓词 “学生x选修了课程y” 

等价变换:

4a5a3fc5c47245189901b82e9bfecaf3.png

变换后语义:不存在这样的课程y,学生201215122选修了y,而学生x没有选。

用NOT EXISTS谓词表示:     SELECT DISTINCT SnoFROM SC SCXWHERE NOT EXISTS(SELECT *FROM SC SCYWHERE SCY.Sno = ' 201215122 '  ANDNOT EXISTS(SELECT *FROM SC SCZWHERE SCZ.Sno=SCX.Sno ANDSCZ.Cno=SCY.Cno));

这篇关于数据库系统概论(超详解!!!) 第四节 关系数据库标准语言SQL(Ⅲ)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/848145

相关文章

Mysql数据库聚簇索引与非聚簇索引举例详解

《Mysql数据库聚簇索引与非聚簇索引举例详解》在MySQL中聚簇索引和非聚簇索引是两种常见的索引结构,它们的主要区别在于数据的存储方式和索引的组织方式,:本文主要介绍Mysql数据库聚簇索引与非... 目录前言一、核心概念与本质区别二、聚簇索引(Clustered Index)1. 实现原理(以 Inno

sqlserver、mysql、oracle、pgsql、sqlite五大关系数据库的对象名称和转义字符

《sqlserver、mysql、oracle、pgsql、sqlite五大关系数据库的对象名称和转义字符》:本文主要介绍sqlserver、mysql、oracle、pgsql、sqlite五大... 目录一、转义符1.1 oracle1.2 sqlserver1.3 PostgreSQL1.4 SQLi

使用python生成固定格式序号的方法详解

《使用python生成固定格式序号的方法详解》这篇文章主要为大家详细介绍了如何使用python生成固定格式序号,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... 目录生成结果验证完整生成代码扩展说明1. 保存到文本文件2. 转换为jsON格式3. 处理特殊序号格式(如带圈数字)4

MySQL数据库双机热备的配置方法详解

《MySQL数据库双机热备的配置方法详解》在企业级应用中,数据库的高可用性和数据的安全性是至关重要的,MySQL作为最流行的开源关系型数据库管理系统之一,提供了多种方式来实现高可用性,其中双机热备(M... 目录1. 环境准备1.1 安装mysql1.2 配置MySQL1.2.1 主服务器配置1.2.2 从

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

MyBatis常用XML语法详解

《MyBatis常用XML语法详解》文章介绍了MyBatis常用XML语法,包括结果映射、查询语句、插入语句、更新语句、删除语句、动态SQL标签以及ehcache.xml文件的使用,感兴趣的朋友跟随小... 目录1、定义结果映射2、查询语句3、插入语句4、更新语句5、删除语句6、动态 SQL 标签7、ehc

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

Python版本信息获取方法详解与实战

《Python版本信息获取方法详解与实战》在Python开发中,获取Python版本号是调试、兼容性检查和版本控制的重要基础操作,本文详细介绍了如何使用sys和platform模块获取Python的主... 目录1. python版本号获取基础2. 使用sys模块获取版本信息2.1 sys模块概述2.1.1