【数据库】数据库入门(四): SQL查询 - SELETE的进阶使用
集合操作
常用的集合操作主要有三种:UNION(联合集)、INTERSECT(交叉集)、EXCEPT(求差集)。以上三种集合的操作都是直接作用在两个或者多个 SQL 查询语句之间,将所有的元组按照特定的要求筛选后拼接起来。SQL 查询后实际上是得到一个新的数据表的形式,因此所作用的数据表之间必须定义相同的属性,且属性定义的顺序相同。
举个例子,要获取所有拥有 gmail 或者 hotmail 邮箱账号的学生信息:
(SELECT * FROM Student WHERE Email like '%@gmail.com')
UNION
(SELECT * FROM Student WHERE Email like '%@hotmail.com');
但以下语句无法正常工作,因为所作用的两个表的属性不同:
(SELECT StudentID, Name FROM Student)
UNION
(SELECT Email FROM Student);
连接操作 JOIN
当我们想要查询得到的数据包含多个表的内容,也就是要同时获得不同表中属性数值的时候,常使用 JOIN 语句
内连接(INNER JOIN)
内连接是应用程序中用的普遍的"连接"操作,它一般都是默认连接类型。内连接基于连接谓词将两张表(如 A 和 B)的列组合在一起,产生新的结果表。查询会将 A 表的每一行和 B 表的每一行进行比较,并找出满足连接谓词的组合。当连接谓词被满足,A 和 B 中匹配的行会按列组合(并排组合)成结果集中的一行。
内连接可以进一步被分为:相等连接、自然连接、交叉连接
交叉连接 (CROSS JOIN)
交叉连接,又称为笛卡尔连接(Cartesian join)或者叉乘(Product),它是所有类型内连接的基础 。把表视为行记录的集合,交叉连接返回这两个集合的笛卡尔积。等价于连接条件永远为“真”。
如果 A 和 B 是两个集合,那么它们的交叉连接就记为:A x B。比如:
SELECT * FROM employee CROSS JOIN department; -- 隐式表达交叉连接
SELECT * FROM employee, department;
相等连接(equi-join,或 equijoin)
相等连接是比较连接的一种特例,它的连接谓词只用了相等比较。通常在 JOIN 语句后跟 ON 关键字补充相等比较语句。举个例子,找出所有至少存在一名学生注册的课程名字:
SELECT DISTINCT c.Cname
FROM Course c INNER JOIN Enrol e ON c.No=e.CourseNo;
自然连接(NATURAL JOIN)
自然连接比相等连接的进一步特例化。两表做自然连接时,两表中的所有名称相同的列都将被比较,这是隐式的。自然连接得到的结果表中,两表中名称相同的列只出现一次。也就是说,通过相等连接得到的结果表中,用于比较的列可能会重复,而在自然连接中并不会出现。
SELECT *
FROM Student s NATURAL JOIN Enrol1 e;
外连接(OUTER JOIN)
外连接与内连接最大的区别在于,外连接并不要求连接的两表的每一条记录在对方表中都一条匹配的记录。要保留所有记录(甚至这条记录没有匹配的记录也要保留)的表称为保留表,没有匹配的部分用 NULL 代替。
外连接可依据连接表保留左表,右表或全部表的行而进一步分为左外连接,右外连接和全连接。
需要注意一点,当外部连接既包含 ON 子句又包含 WHERE 子句时,应当只把表之间的连接条件写在 ON 子句中,对表中数据的筛选必须写在 WHERE 子句中。而内部连接的各条件表达式既可以放在 ON 子句又可以放在 WHERE 子句中。这是因为对于外部连接,保留表中被 ON 子句筛除掉的行要被添加回来,在此操作之后才会用 WHERE 子句去筛选连接结果中的各行。
左外连接(LEFT OUTER JOIN)
左外连接(left outer join),亦简称为左连接(left join),若 A 和 B 两表进行左外连接,那么结果表中将包含"左表"(即表 A)的所有记录,即使那些记录在"右表" B 没有符合连接条件的匹配。这意味着即使 ON 语句在 B 中的匹配项是0条,连接操作还是会返回一条记录,只不过这条记录中来自于 B 的每一列的值都为 NULL。这意味着左外连接会返回左表的所有记录和右表中匹配记录的组合(如果右表中无匹配记录,来自于右表的所有列的值设为 NULL)。如果左表的一行在右表中存在多个匹配行,那么左表的行会复制和右表匹配行一样的数量,并进行组合生成连接结果。
SELECT *
FROM Student s LEFT JOIN Enrol1 e
ON s.StudentID=e.StudentID;
右外连接(RIGHT OUTER JOIN)
右外连接(right outer join),亦简称右连接(right join),它与左外连接完全类似,只不过是作连接的表的顺序相反而已。如果 A 表右连接 B 表,那么"右表" B 中的每一行在连接表中至少会出现一次。如果 B 表的记录在"左表" A 中未找到匹配行,连接表中来源于 A 的列的值设为 NULL。
右连接操作返回右表的所有行和这些行在左表中匹配的行(没有匹配的,来源于左表的列值设为 NULL)。
SELECT *
FROM Student s RIGHT JOIN Enrol1 e
ON s.StudentID=e.StudentID;
子查询 Subqueries
子查询通常用在 FROM 子句或者 WHERE 子句当中。用于 FROM 子句的目的通常是从查询的到结果表中采取进一步的查询,比如调用内置函数,获取最大值、最小值、平均值、计数等等。用于 WHERE 子句的情况,通常会使用一下关键字:
- IN:检查某个元组是否存在于子查询结果中。
- EXISTS:检查子查询结果是否为空。
- ALL, SOME or ANY: 使用子查询结果前对该结果进行相应的比较。
- NOT:上述关键字前使用 NOT 是结果取反。
列出最少有10名学生参加的课程的所有学生信息以及该课程的名字:
SELECT s.*, e1.CourseNo
FROM Student s NATURAL JOIN Enrol e1
WHERE e1.CourseNo IN
(SELECT e2.CourseNo
FROM Enrol e2
GROUP BY e2.CourseNo
HAVING COUNT(*) < 10);
列出所有至少参加一门课程的学生信息:
SELECT s.*
FROM Student s
WHERE EXISTS (SELECT *
FROM Enrol e
WHERE s.StudentID=e.StudentID);
列出所有没有参加任何课程的学生信息:
SELECT s.*
FROM Student s
WHERE NOT EXISTS (SELECT *
FROM Enrol e
WHERE s.StudentID=e.StudentID);
列出在2016学年第二学期中,报名学生人数最多的课程:
SELECT e.CourseNo
FROM (SELECT e1.CourseNo, COUNT(*) AS NoOfStudents
FROM Enrol e1
WHERE e1.Semester = '2016 S2'
GROUP BY e1.CourseNo) e
WHERE e.NoOfStudents =
(SELECT MAX(e2.NoOfStudents)
FROM (SELECT e1.CourseNo, COUNT(*) AS NoOfStudents
FROM Enrol e1
WHERE e1.Semester = '2016 S2'
GROUP BY e1.CourseNo) e2);
列出在2016学年第二学期中,报名学生人数比至少一个其他课程要多的课程:
SELECT e.CourseNo
FROM (SELECT e1.CourseNo, COUNT(*) AS NoOfStudents
FROM Enrol e1
WHERE e1.Semester = '2016 S2'
GROUP BY e1.CourseNo) e
WHERE e.NoOfStudents > ANY
(SELECT e2.NoOfStudents
FROM (SELECT e1.CourseNo, COUNT(*) AS NoOfStudents
FROM Enrol e1
WHERE e1.Semester = '2016 S2'
GROUP BY e1.CourseNo) e2);
具体创建数据库的代码可以通过下面的链接得到:
https://github.com/OddUlrich/Experiment-Code/tree/master/SQL%20Select%20Practice
若想做更多关于查询的练习,可以访问下面的这个网站:
https://pgexercises.com/
参考资料:
连接 (join):https://zh.wikipedia.org/wiki/%E8%BF%9E%E6%8E%A5
【数据库】数据库入门(四): SQL查询 - SELETE的进阶使用的更多相关文章
- Sql Server 导出数据库表结构的SQL查询语句
--导出数据库所有表 SELECT 表名 Then D.name Else '' End, 表说明 Then isnull(F.value,'') Else '' End, 字段序号 = A.colo ...
- SQL查询语句的进阶使用
MySQL的进阶使用 sql语句一些功能的使用 导入现有大量数据文件步骤 1) 把*.sql文件拷贝到Linux某一位置(例如Desktop) 2) Linux命令行进入该位置 cd ~/Deskto ...
- 浏览器数据库 IndexedDB 入门
一.概述 随着浏览器的功能不断增强,越来越多的网站开始考虑,将大量数据储存在客户端,这样可以减少从服务器获取数据,直接从本地获取数据. 现有的浏览器数据储存方案,都不适合储存大量数据:Cookie 的 ...
- 浏览器数据库 IndexedDB 入门教程
一.概述 随着浏览器的功能不断增强,越来越多的网站开始考虑,将大量数据储存在客户端,这样可以减少从服务器获取数据,直接从本地获取数据. 现有的浏览器数据储存方案,都不适合储存大量数据:Cookie 的 ...
- SQL查询语句大全集锦
SQL查询语句大全集锦 一. 简单查询 简单的Transact-SQL查询只包括选择列表.FROM子句和WHERE子句.它们分别说明所查询列.查询的 表或视图.以及搜索条件等. 例如,下面的语句查询t ...
- 利用SQL查询扶贫对象医保报销比率的审计方法
利用SQL查询扶贫对象医保报销比率的审计方法 扶贫资金惠及贫困百姓的切身利益,主管部门多,资金实行逐级下拨,并且扶贫项目小而分散,主要在乡镇和农村实施.根据湖北省审计厅关于2017年扶贫审计工作方案的 ...
- 笔记:Hibernate SQL 查询
Hibernate 支持使用原生的SQL查询,使用原生SQL查询可以利用某些数据库特性,原生SQL查询也支持将SQL语句放在配置文件中配置,从而提高程序的解耦,命名SQL查询还可以用于调用存储过程. ...
- Hibernate SQL 查询
本文转载自:https://www.cnblogs.com/li3807/p/6358386.html Hibernate 支持使用原生的SQL查询,使用原生SQL查询可以利用某些数据库特性,原生SQ ...
- SQL Server Extended Events 进阶 1:从SQL Trace 到Extended Events
http://www.sqlservercentral.com/articles/Stairway+Series/134869/ SQL server 2008 中引入了Extended Events ...
随机推荐
- Tkinter--Text文本框样例
#-*- coding:utf-8 -*- """ Text 文本框样例 实现功能有:Ctrl+a全选文本, 竖向滚动条,横向滚动条(不自动换行) 自动缩放 有谁知道全选 ...
- 【oracle】update select语句
- 显示隐藏文件.reg
显示隐藏文件.reg Windows Registry Editor Version 5.00 [HKEY_CURRENT_USER\Software\Microsoft\Windows\Curren ...
- Linux性能优化实战学习笔记:第五十三讲
一.上节回顾 在前面的内容中,我为你介绍了很多性能分析的原理.思路以及相关的工具.不过,在实际的性能分析中,一个很常见的现象是,明明发生了性能瓶颈,但当你登录到服务器中想要排查的时候,却发现瓶颈已经消 ...
- OBS录制全屏游戏的方法(超好录屏)
新版Windows设置 详见 https://github.com/obsproject/obs-studio/wiki/Laptop-Troubleshooting 新版的Windows 10: l ...
- 本地搭建WordPress (XAMPP环境)
1,XAMPP是一个流行的PHP开发环境,官网下载: https://www.apachefriends.org/zh_cn/index.html 然后安装. 官方介绍:XAMPP是最流行的PHP开发 ...
- 第01组 Beta冲刺(5/5)
队名:007 组长博客: https://www.cnblogs.com/Linrrui/p/12031875.html 作业博客: https://edu.cnblogs.com/campus/fz ...
- [转载]3.15 UiPath图片操作保存图片的介绍和使用
一.保存图像 (Save Image)的介绍 可以将图像保存到磁盘的一种活动 二.保存图像 (Save Image)在UiPath中的使用 1.打开设计器,在设计库中新建一个Sequence,为序列命 ...
- Android studio(AS) svg图片使用
1.下载svg文件,https://www.iconfont.cn/阿里的2.通过AS - New - Vectro Asset转换成安卓可用的xml文件(ic_back.xml)3.配置Gradle ...
- [转帖]k8s 如何让你的应用活的更久
k8s 如何让你的应用活的更久 https://www.jianshu.com/p/132319e795ae 众所周知,k8s 可以托管你的服务 / 应用,当出现各种原因导致你的应用挂掉之后,k8s ...