数据库系统概论(超详解!!!) 第四节 关系数据库标准语言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中的group by高级用法

《mysql中的groupby高级用法》MySQL中的GROUPBY是数据聚合分析的核心功能,主要用于将结果集按指定列分组,并结合聚合函数进行统计计算,下面给大家介绍mysql中的groupby用法... 目录一、基本语法与核心功能二、基础用法示例1. 单列分组统计2. 多列组合分组3. 与WHERE结合使

Nginx location匹配模式与规则详解

《Nginxlocation匹配模式与规则详解》:本文主要介绍Nginxlocation匹配模式与规则,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、环境二、匹配模式1. 精准模式2. 前缀模式(不继续匹配正则)3. 前缀模式(继续匹配正则)4. 正则模式(大

Android实现在线预览office文档的示例详解

《Android实现在线预览office文档的示例详解》在移动端展示在线Office文档(如Word、Excel、PPT)是一项常见需求,这篇文章为大家重点介绍了两种方案的实现方法,希望对大家有一定的... 目录一、项目概述二、相关技术知识三、实现思路3.1 方案一:WebView + Office Onl

Mysql用户授权(GRANT)语法及示例解读

《Mysql用户授权(GRANT)语法及示例解读》:本文主要介绍Mysql用户授权(GRANT)语法及示例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录mysql用户授权(GRANT)语法授予用户权限语法GRANT语句中的<权限类型>的使用WITH GRANT

Mysql如何解决死锁问题

《Mysql如何解决死锁问题》:本文主要介绍Mysql如何解决死锁问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录【一】mysql中锁分类和加锁情况【1】按锁的粒度分类全局锁表级锁行级锁【2】按锁的模式分类【二】加锁方式的影响因素【三】Mysql的死锁情况【1

Java实现优雅日期处理的方案详解

《Java实现优雅日期处理的方案详解》在我们的日常工作中,需要经常处理各种格式,各种类似的的日期或者时间,下面我们就来看看如何使用java处理这样的日期问题吧,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言一、日期的坑1.1 日期格式化陷阱1.2 时区转换二、优雅方案的进阶之路2.1 线程安全重构2

Java中的JSONObject详解

《Java中的JSONObject详解》:本文主要介绍Java中的JSONObject详解,需要的朋友可以参考下... Java中的jsONObject详解一、引言在Java开发中,处理JSON数据是一种常见的需求。JSONObject是处理JSON对象的一个非常有用的类,它提供了一系列的API来操作J

HTML5中的Microdata与历史记录管理详解

《HTML5中的Microdata与历史记录管理详解》Microdata作为HTML5新增的一个特性,它允许开发者在HTML文档中添加更多的语义信息,以便于搜索引擎和浏览器更好地理解页面内容,本文将探... 目录html5中的Mijscrodata与历史记录管理背景简介html5中的Microdata使用M

html5的响应式布局的方法示例详解

《html5的响应式布局的方法示例详解》:本文主要介绍了HTML5中使用媒体查询和Flexbox进行响应式布局的方法,简要介绍了CSSGrid布局的基础知识和如何实现自动换行的网格布局,详细内容请阅读本文,希望能对你有所帮助... 一 使用媒体查询响应式布局        使用的参数@media这是常用的

HTML5表格语法格式详解

《HTML5表格语法格式详解》在HTML语法中,表格主要通过table、tr和td3个标签构成,本文通过实例代码讲解HTML5表格语法格式,感兴趣的朋友一起看看吧... 目录一、表格1.表格语法格式2.表格属性 3.例子二、不规则表格1.跨行2.跨列3.例子一、表格在html语法中,表格主要通过< tab