集合操作
常用的集合操作主要有三种: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的进阶使用的更多相关文章

  1. Sql Server 导出数据库表结构的SQL查询语句

    --导出数据库所有表 SELECT 表名 Then D.name Else '' End, 表说明 Then isnull(F.value,'') Else '' End, 字段序号 = A.colo ...

  2. SQL查询语句的进阶使用

    MySQL的进阶使用 sql语句一些功能的使用 导入现有大量数据文件步骤 1) 把*.sql文件拷贝到Linux某一位置(例如Desktop) 2) Linux命令行进入该位置 cd ~/Deskto ...

  3. 浏览器数据库 IndexedDB 入门

    一.概述 随着浏览器的功能不断增强,越来越多的网站开始考虑,将大量数据储存在客户端,这样可以减少从服务器获取数据,直接从本地获取数据. 现有的浏览器数据储存方案,都不适合储存大量数据:Cookie 的 ...

  4. 浏览器数据库 IndexedDB 入门教程

    一.概述 随着浏览器的功能不断增强,越来越多的网站开始考虑,将大量数据储存在客户端,这样可以减少从服务器获取数据,直接从本地获取数据. 现有的浏览器数据储存方案,都不适合储存大量数据:Cookie 的 ...

  5. SQL查询语句大全集锦

    SQL查询语句大全集锦 一. 简单查询 简单的Transact-SQL查询只包括选择列表.FROM子句和WHERE子句.它们分别说明所查询列.查询的 表或视图.以及搜索条件等. 例如,下面的语句查询t ...

  6. 利用SQL查询扶贫对象医保报销比率的审计方法

    利用SQL查询扶贫对象医保报销比率的审计方法 扶贫资金惠及贫困百姓的切身利益,主管部门多,资金实行逐级下拨,并且扶贫项目小而分散,主要在乡镇和农村实施.根据湖北省审计厅关于2017年扶贫审计工作方案的 ...

  7. 笔记:Hibernate SQL 查询

    Hibernate 支持使用原生的SQL查询,使用原生SQL查询可以利用某些数据库特性,原生SQL查询也支持将SQL语句放在配置文件中配置,从而提高程序的解耦,命名SQL查询还可以用于调用存储过程. ...

  8. Hibernate SQL 查询

    本文转载自:https://www.cnblogs.com/li3807/p/6358386.html Hibernate 支持使用原生的SQL查询,使用原生SQL查询可以利用某些数据库特性,原生SQ ...

  9. SQL Server Extended Events 进阶 1:从SQL Trace 到Extended Events

    http://www.sqlservercentral.com/articles/Stairway+Series/134869/ SQL server 2008 中引入了Extended Events ...

随机推荐

  1. 配置 application.properties

    # 数据库链接信息mysql.driver=com.mysql.cj.jdbc.Drivermysql.url=jdbc:mysql://localhost:3306/mydemo?character ...

  2. USACO Buying Hay

    洛谷 P2918 [USACO08NOV]买干草Buying Hay https://www.luogu.org/problem/P2918 JDOJ 2592: USACO 2008 Nov Sil ...

  3. Linux性能优化实战学习笔记:第十三讲

    问题1:性能工具版本太低,导致指标不全 解决方案1: 这是使用 CentOS 的同学普遍碰到的问题.在文章中,我的pidstat 输出里有一个 %wait 指标,代表进程等待 CPU 的时间百分比, ...

  4. [LeetCode] 834. Sum of Distances in Tree 树中距离之和

    An undirected, connected tree with N nodes labelled 0...N-1 and N-1 edges are given. The ith edge co ...

  5. [LeetCode] 415. Add Strings 字符串相加

    Given two non-negative numbers num1 and num2 represented as string, return the sum of num1 and num2. ...

  6. K8s 集群安装(一)

    01,集群环境 三个节点   master node1 node2 IP 192.168.0.81 192.168.0.82 192.168.0.83 环境 centos 7 centos 7 cen ...

  7. http & https & http/2 & SPDY & WebSocket

    参考文章: http https :https://www.jianshu.com/p/d286d097e56b https & ssl:https://www.jianshu.com/p/2 ...

  8. CentOS7 Harbor 安装

    安装Harbor,首先要安装docker 和 docker-compose 1.安装docker   (1)安装一些必要的系统工具 $ yum install -y yum-utils device- ...

  9. 论文阅读: Building a 3-D Line-Based Map Using Stereo SLAM

    Abstract 一个把直线用作feature的SLAM系统. 跟点相比, 直线对于环境的结构提供了更丰富的信息, 也让其鞥有可能推断地图的空间语义. 使用了Plucker line coordian ...

  10. 怎么写自己的CMakeLists.txt--二

    之前写过一篇及其简单的关于CMakeLists.txt的写法,现在重点剖析find_package的用法. 如果程序中使用了外部库,事先并不知道它的头文件和链接库的位置,就要给出头文件和链接库的查找方 ...