以初学者的角度理解:SQL实现关系除法

相信各位在学习SQL的时候,由于没有一家SQL语言提供除法命令而只能自己写一个。而网上大多就是四步骤加一个模板:

select distinct	A.X
from A A1
where not exists(
select B.Y
from B
where not exists(
select *
from A A2
where A1.X = A2.X
and A2.Y = B.Y
)
)

那四个步骤又写的过于抽象~,看得一头雾水。因此笔者希望从一个初学者的角度,讲解一下关系除法的实现过程,帮助大家理解。

例子

我们举一个实例来讲解~

我们用一张SC表和Course表,其中:

SC表:

Course表:

而我们想要做的是:

查询选修了全部课程的学生姓名。

很明显我们需要做除法:

select distinct	SC1.Sno
from SC SC1
where not exists(
select Course.Cno
from Course
where not exists(
select *
from SC SC2
where SC1.Sno = SC2.Sno
and SC2.Cno = Course.Cno
)
)

接下来讲解这段语句的整个过程~

具体讲解

我们先看看这两句SQL

select distinct	SC1.Sno
from SC SC1

这一段,产生的结果应该是:

接下来到下一句:

where			not exists()

这句话的意思很简单,就是我即将进行括号中写好的查询语句,如果查询结果为空,返回true,否则返回false

这时候,我们开始where语句,此时SC1.Sno的值会被挨个放进where中进行处理。

此时:

接下来进入not exists的部分。

select			Course.Cno
from Course

跟上面一样,先看看它执行出来什么效果:

接下来到下一句:

where			not exists()

此时:

接下来进入下一个not exists的部分。

select			*
from SC SC2
where SC1.Sno = SC2.Sno
and SC2.Cno = Course.Cno

我们的SC表是这样子的:

我们看看当select完之后,where在干嘛:

where			SC1.Sno = SC2.Sno
and SC2.Cno = Course.Cno

这句话的意思是:

SC1.Sno与本关系中的Sno相等且Course.Cno与本关系中Cno的值相等

那回头看看,我们进行的步骤,我们发现到这一步的时候,SC1.Sno = 20110001,而Course.Cno = 001。换言之,我们要找到符合这两个条件的元组。

不难找到:

因此

select			*
from SC SC2
where SC1.Sno = SC2.Sno
and SC2.Cno = Course.Cno

返回结果集:

20110001 001 89

这时候,我们看回上一层结构:

select			Course.Cno
from Course
where not exists(...)

我们成功返回了not exists部分的结果集,因此where处得到的结果是false,因此Course.Cno = 001不被加入这层结构产生的结果集。所以指针往下,Course.Cno的数据变更:

Coures.Cno = 002

于是我们再次进入该语句:

select			*
from SC SC2
where SC1.Sno = SC2.Sno
and SC2.Cno = Course.Cno

此时SC1.Sno = 20110001,而Course.Cno = 002

返回结果集:

20110001 002 78

这时候,我们看回上一层结构:

select			Course.Cno
from Course
where not exists(...)

我们成功返回了not exists部分的结果集,因此where处得到的结果是false,因此Course.Cno = 002不被加入这层结构产生的结果集。所以指针往下,Course.Cno的数据变更:

Coures.Cno = 003

于是我们再次进入该语句:

select			*
from SC SC2
where SC1.Sno = SC2.Sno
and SC2.Cno = Course.Cno

此时SC1.Sno = 20110001,而Course.Cno = 003

返回结果集:

20110001 003 89

这时候,我们看回上一层结构:

select			Course.Cno
from Course
where not exists(...)

我们成功返回了not exists部分的结果集,因此where处得到的结果是false,因此Course.Cno = 002不被加入这层结构产生的结果集。所以指针往下,Course.Cno的数据变更:

Coures.Cno = 004

重复上述的操作,不再赘述。

我们发现,由于20110001这位同学刚刚好都把课选了。每次我们都能从SC表中找到数据,因此

select			Course.Cno
from Course
where not exists(...)

该段语句最终的结果是一个空集,没有一个数据被放进了结果集。

这时候我们再回到上一层结构:

select distinct	SC1.Sno
from SC SC1
where not exists(...)

该处的not exists中的查询语句返回的是空集,所以它返回的是true

也就是说,where处得到的结果是true

也就是说,Sno = 20110001 被加入结果集。

我们继续分析,想要Sno不被加入结果集:

select			Course.Cno
from Course
where not exists(...)

这段语句就需要返回结果集。

也就是说这段语句not exists部分至少有一个返回true

那么这段语句:

select			*
from SC SC2
where SC1.Sno = SC2.Sno
and SC2.Cno = Course.Cno

只要元祖没有被找到(即有一门课该学生没有选),它就返回空集。

接下来一系列的连锁反应:

select			Course.Cno
from Course
where not exists(...)

返回一个非空结果集

select distinct	SC1.Sno
from SC SC1
where not exists(...)

由于返回的是true,所以该学生的学号不被加入结果集。

总结一下

select distinct	A.X
from A A1
where not exists(
select B.Y
from B
where not exists(
select *
from A A2
where A1.X = A2.X
and A2.Y = B.Y
)
)

实际上,直观来看:

select distinct	A.X
from A A1
where not exists()

这一段相当于按X分组。

select			B.Y
from B
where not exists(
select *
from A A2
where A1.X = A2.X
and A2.Y = B.Y
)

这段就是分组来看,去确定在一组中,是不是有某一个数据没有没有在B.Y中,如果有,返回没有在B.Y中的结果

如果的确有一个数据没在其中,由于它是嵌套在not exists中的,只要它返回没有在B.Y中的结果不是空集,就说明该组不合要求,不加入结果集。如果是空集,说明符合要求,加入结果集。

瞎说一下

我知道非常绕,静下心按流程走一走,还是看不懂可以留言喔~

以初学者的角度理解:SQL实现关系除法的更多相关文章

  1. 转:如何学习SQL(第二部分:从关系角度理解SQL)

    转自:http://blog.163.com/mig3719@126/blog/static/285720652010950825538/ 6. 从关系角度理解SQL 6.1. 关系和表 众所周知,我 ...

  2. 0419如何利用关系角度看待SQL

    转自http://www.open-open.com/solution/view/1389339225820 十步完全理解SQL   1. SQL 是一种声明式语言 首先要把这个概念记在脑中:“声明” ...

  3. 【转载】十步完全理解SQL

    很多程序员视 SQL 为洪水猛兽.SQL 是一种为数不多的声明性语言,它的运行方式完全不同于我们所熟知的命令行语言.面向对象的程序语言.甚至是函数语言(尽管有些人认为 SQL 语言也是一种函数式语言) ...

  4. 转载文章----十步完全理解SQL

    转载地址:http://blog.jobbole.com/55086/ 很多程序员视 SQL 为洪水猛兽.SQL 是一种为数不多的声明性语言,它的运行方式完全不同于我们所熟知的命令行语言.面向对象的程 ...

  5. 十步完全理解SQL

    转载于:http://blog.jobbole.com/55086/ 很多程序员视 SQL 为洪水猛兽.SQL 是一种为数不多的声明性语言,它的运行方式完全不同于我们所熟知的命令行语言.面向对象的程序 ...

  6. [转]十步完全理解SQL

    原文地址:http://blog.jobbole.com/55086/ 很多程序员视 SQL 为洪水猛兽.SQL 是一种为数不多的声明性语言,它的运行方式完全不同于我们所熟知的命令行语言.面向对象的程 ...

  7. 十步完全理解 SQL(转载)

    英文出处:Lukas Eder. 很多程序员视 SQL 为洪水猛兽.SQL 是一种为数不多的声明性语言,它的运行方式完全不同于我们所熟知的命令行语言.面向对象的程序语言.甚至是函数语言(尽管有些人认为 ...

  8. 经典:十步完全理解 SQL

    经典:十步完全理解 SQL   来源:伯乐在线 链接:http://blog.jobbole.com/55086/ 很多程序员视 SQL 为洪水猛兽.SQL 是一种为数不多的声明性语言,它的运行方式完 ...

  9. 简单十步让你全面理解SQL

    很多程序员认为SQL是一头难以驯服的野兽.它是为数不多的声明性语言之一,也因为这样,其展示了完全不同于其他的表现形式.命令式语言. 面向对象语言甚至函数式编程语言(虽然有些人觉得SQL 还是有些类似功 ...

随机推荐

  1. 前台JS遍历Table的所有单元格数据内容

    在日常开发过程中为了减少与后台服务器的交互次数,大部分的功能都会放到前台使用JS来完成. 例如:一个表中有ID(FOCUS_SEQ)和Name(COLUMNCTRL)两个字段,其中ID是自定义连续增长 ...

  2. Spring Cloud系列(五):服务网关Zuul

    在前面的篇章都是一个服务消费者去调用一个服务提供者,但事实上我们的系统基本不会那么简单,如果真的是那么简单的业务架构我们也没必要用Spring Cloud,直接部署一个Spring Boot应用就够了 ...

  3. 阿里面试挂了,就因为面试官说我Spring 事务管理(器)不熟练?

    前言 事务管理,一个被说烂的也被看烂的话题,还是八股文中的基础股之一.但除了八股文中需要熟读并背诵的那些个传播行为之外,背后的"为什么"和核心原理更为重要. ​ 写这篇文章之前,我 ...

  4. 单点突破:MySQL之索引

    前言 开发环境:MySQL5.7.31 什么是索引 在MySQL中,索引(Index)是帮助高效获取数据的数据结构. 我们可以将数据库理解为一本书,数据库中的各个数据列(column)就是目录中的章节 ...

  5. CentOS 30分钟部署 .net core 在线客服系统

    前段时间我发表了一系列文章,开始介绍基于 .net core 的在线客服系统开发过程.期间有一些朋友希望能够给出 Linux 环境的安装部署指导,本文基于 CentOS 8.3 来安装部署.在本文中我 ...

  6. 搭建支持SSL加密传输的vftpd

    让vsftpd支持SSL 必须让OPENSSL≥0.9.6版本还有就是本身vsftpd版本是否支持 查询vsftpd软件是否支持SSL        [root@localhost vsftpd]# ...

  7. 【VBA】查找字符串

    老婆饼里有老婆吗 Sub test() aaa = "老婆饼里有老婆吗" If InStr(aaa, "老婆") <> 0 Then Debug.p ...

  8. MySQL:聊一聊数据库中的那些锁

    在软件开发中,程序在高并发的情况下,为了保证一致性或者说安全性,我们通常都会通过加锁的方式来解决,在 MySQL 数据库中同样有这样的问题,一方面为了最大程度的利用数据库的并发访问,另一方面又需要保证 ...

  9. hackthebox TheNotebook

    前言 只拿到了user,提权没成功--有wp说是CVE-2019-5736,我没打成. 打点 nmap-sV -v -A 10.10.10.230 端口扫描结果: PORT STATE SERVICE ...

  10. vue cli3 使用elemet-plus

    原文章: https://blog.csdn.net/qq_44827865/article/details/115457445 element-plus官方网站:https://element-pl ...