1.简单查询

去重

  1. SELECT DISTINCT 列名 FROM 表名;
  1. mysql> SELECT department FROM student_info;
  2. +-----------------+
  3. | department |
  4. +-----------------+
  5. | 计算机学院 |
  6. | 计算机学院 |
  7. | 计算机学院 |
  8. | 计算机学院 |
  9. | 航天学院 |
  10. | 航天学院 |
  11. +-----------------+
  12. 6 rows in set (0.00 sec)
  13. mysql> SELECT DISTINCT department FROM student_info;
  14. +-----------------+
  15. | department |
  16. +-----------------+
  17. | 计算机学院 |
  18. | 航天学院 |
  19. +-----------------+
  20. 2 rows in set (0.00 sec)

同时针对多列去重

  1. SELECT DISTINCT 列名1, 列名2, ... 列名n FROM 表名;
  1. mysql> SELECT department, major FROM student_info;
  2. +-----------------+--------------------------+
  3. | department | major |
  4. +-----------------+--------------------------+
  5. | 计算机学院 | 计算机科学与工程 |
  6. | 计算机学院 | 计算机科学与工程 |
  7. | 计算机学院 | 软件工程 |
  8. | 计算机学院 | 软件工程 |
  9. | 航天学院 | 飞行器设计 |
  10. | 航天学院 | 电子信息 |
  11. +-----------------+--------------------------+
  12. 6 rows in set (0.00 sec)
  13. mysql> SELECT DISTINCT department, major FROM student_info;
  14. +-----------------+--------------------------+
  15. | department | major |
  16. +-----------------+--------------------------+
  17. | 计算机学院 | 计算机科学与工程 |
  18. | 计算机学院 | 软件工程 |
  19. | 航天学院 | 飞行器设计 |
  20. | 航天学院 | 电子信息 |
  21. +-----------------+--------------------------+
  22. 4 rows in set (0.00 sec)
  23. mysql>

限制查询结果条数

  1. - 开始行指的是我们想从第几行数据开始查询(从0开始计数),限制条数是结果集中最多包含多少条记录。
  2. - LIMIT后边也可以只有一个参数,那这个参数就代表着限制行数。也就是说我们可以不指定开始行,默认的开始行就是第0
  1. LIMIT 开始行, 限制条数;
  1. mysql> SELECT number, name, id_number, major FROM student_info LIMIT 0, 2;
  2. +----------+-----------+--------------------+--------------------------+
  3. | number | name | id_number | major |
  4. +----------+-----------+--------------------+--------------------------+
  5. | 20180101 | 杜子腾 | 158177199901044792 | 计算机科学与工程 |
  6. | 20180102 | 杜琦燕 | 151008199801178529 | 计算机科学与工程 |
  7. +----------+-----------+--------------------+--------------------------+
  8. 2 rows in set (0.00 sec)
  9. mysql>
  1. - 如果指定的开始行大于结果中的行数,那查询结果就什么都没有:
  1. mysql> SELECT number, name, id_number, major FROM student_info LIMIT 6, 2;
  2. Empty set (0.00 sec)
  3. mysql>
  1. - 如果查询的结果条数不超过限制条数,那就可以全部显式出来

对查询结果排序

  1. - MySQL默认会按照这些数据底层存储的顺序来给我们返回数据,默认不能确定排序顺序
  1. ORDER BY 1 ASC|DESC, 2 ASC|DESC ...
  1. mysql> SELECT * FROM student_score ORDER BY score DESC;
  2. +----------+-----------------------------+-------+
  3. | number | subject | score |
  4. +----------+-----------------------------+-------+
  5. | 20180102 | 母猪的产后护理 | 100 |
  6. | 20180102 | 论萨达姆的战争准备 | 98 |
  7. | 20180101 | 论萨达姆的战争准备 | 88 |
  8. | 20180101 | 母猪的产后护理 | 78 |
  9. | 20180103 | 论萨达姆的战争准备 | 61 |
  10. | 20180103 | 母猪的产后护理 | 59 |
  11. | 20180104 | 母猪的产后护理 | 55 |
  12. | 20180104 | 论萨达姆的战争准备 | 46 |
  13. +----------+-----------------------------+-------+
  14. 8 rows in set (0.00 sec)
  15. mysql>
  1. - 不指定排序方向,则默认使用的是ASC,也就是从小到大的升序规则
  2. - ORDER BY语句和LIMIT语句结合使用,不过 ORDER BY 语句必须放在 LIMIT 语句前边,比如这样:
  1. mysql> SELECT * FROM student_score ORDER BY score LIMIT 1;
  2. +----------+-----------------------------+-------+
  3. | number | subject | score |
  4. +----------+-----------------------------+-------+
  5. | 20180104 | 论萨达姆的战争准备 | 46 |
  6. +----------+-----------------------------+-------+
  7. 1 row in set (0.00 sec)
  8. mysql>

2.条件查询

简单的比较操作符

操作符 示例 描述
= a = b a等于b
<>或者!= a <> b a不等于b
< a < b a小于b
<= a <= b a小于或等于b
> a > b a大于b
>= a >= b a大于或等于b
BETWEEN a BETWEEN b AND c 满足 b <= a <= c
NOT BETWEEN a NOT BETWEEN b AND c 不满足 b <= a <= c

匹配枚举出的元素

操作符 示例 描述
IN a IN (b1, b2, …) a是b1, b2, … 中的某一个
NOT IN a NOT IN (b1, b2, …) a不是b1, b2, … 中的任意一个

匹配NULL值

  1. - NULL代表没有值,意味着你并不知道该列应该填入什么数据,在判断某一列是否为NULL的时候并不能单纯的使用=操作符,而是需要专业判断值是否是NULL的操作符:
操作符 示例 描述
IS NULL a IS NULL a的值是NULL
IS NOT NULL a IS NOT NULL a的值不是NULL

多个搜索条件的查询

AND操作符: 且

  1. mysql> SELECT * FROM student_score WHERE subject = '母猪的产后护理' AND score > 75;
  2. +----------+-----------------------+-------+
  3. | number | subject | score |
  4. +----------+-----------------------+-------+
  5. | 20180101 | 母猪的产后护理 | 78 |
  6. | 20180102 | 母猪的产后护理 | 100 |
  7. +----------+-----------------------+-------+
  8. 2 rows in set (0.00 sec)
  9. mysql>
  1. - 其中的subject = '母猪的产后护理'score > 75是两个搜索条件,我们使用AND操作符把这两个搜索条件连接起来表示只有当两个条件都满足的记录才能被加入到结果集。

OR操作符: 或

AND OR 注意事项

  1. - AND操作符的优先级高于OR操作符,也就是说在判断某条记录是否符合条件时会先检测AND操作符两边的搜索条件

通配符模糊查询

操作符 示例 描述
LIKE a LIKE b a匹配b
NOT LIKE a NOT LIKE b a不匹配b
  1. - %:代表任意多个字符
  2. - _:代表任意单个字符

LIKE或者NOT LIKE操作符只用于字符串匹配。另外,通配符不能代表NULL,如果需要匹配NULL的话,需要使用IS NULL或者IS NOT NULL。

转义通配符

  1. - 如果匹配字符串中需要普通字符'%'或者'_'的话,需要在它们前边加一个反斜杠\来和通配符区分开来
  2. - '\%'代表普通字符'%'
  3. - '\_'代表普通字符'_'
  1. mysql> SELECT number, name, id_number, major FROM student_info WHERE name LIKE '范\_';
  2. Empty set (0.00 sec)
  3. mysql>

3.表达式和函数

操作数

  1. 1. 常数
  2. 1. 常数很好理解,我们平时用到的数字、字符串、时间值什么的都可以被称为常数,它是一个确定的值,比如数字1,字符串'abc',时间值2019-08-16 17:10:43啥的。
  3. 2. 列名
  4. 1. 针对某个具体的表,它的列名可以被当作表达式的一部分,比如对于student_info表来说,numbername都可以作为操作数。
  5. 3. 函数调用
  6. 1. MySQL中有函数的概念,比方说获取当前时间的函数NOW,而在函数后边加个小括号就算是一个函数调用,比如NOW()。
  7. 4. 标量子查询或者行子查询
  8. 1. <br />
  9. 5. 其他表达式
  10. 1. 一个表达式也可以作为一个操作数与另一个操作数来形成一个更复杂的表达式,比方说(假设col是一个列名):
  11. 1. (col - 5) / 3
  12. 1. (1 + 1) * 2 + col * 3

操作符

  1. 1. 算术操作符

操作符 示例 描述
+ a + b 加法
- a - b 减法
* a * b 乘法
/ a / b 除法
DIV a DIV b 除法,取商的整数部分
% a % b 取余
- -a 负号

DIV和/都表示除法操作符,但是DIV只会取商的整数部分,/会保留商的小数部分。比如表达式 2 DIV 3的结果是0,而2 / 3的结果是0.6667

  1. 2. [比较操作符](#WlNE4)

由比较操作符连接而成的表达式也称为布尔表达式,表示真或者假,也可以称为TRUE或者FALSE。比如1 > 3就代表FALSE,3 != 2就代表TRUE

  1. 3. 逻辑操作符
操作符 示例 描述
AND a AND b 只有a和b同时为真,表达式才为真
OR a OR b 只要a或b有任意一个为真,表达式就为真
XOR a XOR b a和b有且只有一个为真,表达式为真

逻辑操作符是用来将多个布尔表达式连接起来

表达式

  1. - 把这些操作数和操作符相互组合起来就可以组成一个表达式
  2. - 放在查询列表中
  3. - 我们前边都是将列名放在查询列表中的(*号代表所有的列名~)。列名只是表达式中超级简单的一种,我们可以将任意一个表达式作为查询列表的一部分来处理,比方说我们可以在查询student_score表时把score字段的数据都加100,就像这样:
  1. mysql> SELECT number, subject, score + 100 FROM student_score;
  2. +----------+-----------------------------+-------------+
  3. | number | subject | score + 100 |
  4. +----------+-----------------------------+-------------+
  5. | 20180101 | 母猪的产后护理 | 178 |
  6. | 20180101 | 论萨达姆的战争准备 | 188 |
  7. | 20180102 | 母猪的产后护理 | 200 |
  8. | 20180102 | 论萨达姆的战争准备 | 198 |
  9. | 20180103 | 母猪的产后护理 | 159 |
  10. | 20180103 | 论萨达姆的战争准备 | 161 |
  11. | 20180104 | 母猪的产后护理 | 155 |
  12. | 20180104 | 论萨达姆的战争准备 | 146 |
  13. +----------+-----------------------------+-------------+
  14. 8 rows in set (0.00 sec)
  15. mysql>
  1. - 其中的numbersubjectscore + 100都是表达式,结果集中的列的名称也将默认使用这些表达式的名称

函数

  1. - MySQL为我们提供了很多所谓的函数, 例: NOW函数用来获取当前的日期和时间。
  2. - 如果我们想使用这些函数,可以在函数名后加一个小括号()就好,表示调用一下这个函数,简称函数调用。比方说NOW()就代表调用NOW函数来获取当前日期和时间。针对某些包含参数的函数,我们也可以在小括号()里将参数填入,比方说UPPER('abc')表示将字符串'abc'转换为大写格式。

文本处理函数

名称 调用示例 示例结果 描述
LEFT LEFT(‘abc123’, 3) abc 给定字符串从左边取指定长度的子串
RIGHT RIGHT(‘abc123’, 3) 123 给定字符串从右边取指定长度的子串
LENGTH LENGTH(‘abc’) 3 给定字符串的长度
LOWER LOWER(‘ABC’) abc 给定字符串的小写格式
UPPER UPPER(‘abc’) ABC 给定字符串的大写格式
LTRIM LTRIM(‘ abc’) abc 给定字符串左边空格去除后的格式
RTRIM RTRIM(‘abc ‘) abc 给定字符串右边空格去除后的格式
SUBSTRING SUBSTRING(‘abc123’, 2, 3) bc1 给定字符串从指定位置截取指定长度的子串
CONCAT CONCAT(‘abc’, ‘123’, ‘xyz’) abc123xyz 将给定的各个字符串拼接成一个新字符串
  1. - 函数调用也算是一种表达式的操作数,它可以和其他操作数用操作符连接起来组成一个表达式来作为查询列表的一部分或者放到搜索条件中. 举例 CONCAT():
  1. mysql> SELECT CONCAT('学号为', number, '的学生在《', subject, '》课程的成绩是:', score) AS 成绩描述 FROM student_score;
  2. +---------------------------------------------------------------------------------------+
  3. | 成绩描述 |
  4. +---------------------------------------------------------------------------------------+
  5. | 学号为20180101的学生在《母猪的产后护理》课程的成绩是:78 |
  6. | 学号为20180101的学生在《论萨达姆的战争准备》课程的成绩是:88 |
  7. | 学号为20180102的学生在《母猪的产后护理》课程的成绩是:100 |
  8. | 学号为20180102的学生在《论萨达姆的战争准备》课程的成绩是:98 |
  9. | 学号为20180103的学生在《母猪的产后护理》课程的成绩是:59 |
  10. | 学号为20180103的学生在《论萨达姆的战争准备》课程的成绩是:61 |
  11. | 学号为20180104的学生在《母猪的产后护理》课程的成绩是:55 |
  12. | 学号为20180104的学生在《论萨达姆的战争准备》课程的成绩是:46 |
  13. +---------------------------------------------------------------------------------------+
  14. 8 rows in set (0.00 sec)
  15. mysql>

日期和时间处理函数

名称 调用示例 示例结果 描述
NOW NOW() 2019-08-16 17:10:43 返回当前日期和时间
CURDATE CURDATE() 2019-08-16 返回当前日期
CURTIME CURTIME() 17:10:43 返回当前时间
DATE DATE(‘2019-08-16 17:10:43’) 2019-08-16 将给定日期和时间值的日期提取出来
DATE_ADD DATE_ADD(‘2019-08-16 17:10:43’, INTERVAL 2 DAY) 2019-08-18 17:10:43 将给定的日期和时间值添加指定的时间间隔
DATE_SUB DATE_SUB(‘2019-08-16 17:10:43’, INTERVAL 2 DAY) 2019-08-14 17:10:43 将给定的日期和时间值减去指定的时间间隔
DATEDIFF DATEDIFF(‘2019-08-16’, ‘2019-08-17’); -1 返回两个日期之间的天数(负数代表前一个参数代表的日期比较小)
DATE_FORMAT DATE_FORMAT(NOW(),’%m-%d-%Y’) 08-16-2019 用给定的格式显示日期和时间

数值处理函数

名称 调用示例 示例结果 描述
ABS ABS(-1) 1 取绝对值
Pi PI() 3.141593 返回圆周率
COS COS(PI()) -1 返回一个角度的余弦
EXP EXP(1) 2.718281828459045 返回e的指定次方
MOD MOD(5,2) 1 返回除法的余数
RAND RAND() 0.7537623539136372 返回一个随机数
SIN SIN(PI()/2) 1 返回一个角度的正弦
SQRT SQRT(9) 3 返回一个数的平方根
TAN TAN(0) 0 返回一个角度的正切


聚集函数(统计函数)

  1. - 如果将上边介绍的那些函数以函数调用的形式放在查询列表中,那么会为表中符合WHERE条件的每一条记录调用一次该函数
函数名 描述
COUNT 返回某列的行数.
( count(*): 返回所有条目数; count(列名):
返回所有列非NULL的条目数 )
MAX 返回某列的最大值
MIN 返回某列的最小值
SUM 返回某列值之和
AVG 返回某列的平均值

聚集函数中DISTINCT的使用

  1. - 默认情况下,上边介绍的聚集函数将计算指定列的所有非NULL数据,如果我们指定的列中有重复数据的话,可以选择使用DISTINCT来过滤掉这些重复数据。
  2. - 比方说我们想查看一下student_info表中存储了多少个专业的学生信息,就可以这么写:
  1. mysql> SELECT COUNT(DISTINCT major) FROM student_info;
  2. +-----------------------+
  3. | COUNT(DISTINCT major) |
  4. +-----------------------+
  5. | 4 |
  6. +-----------------------+
  7. 1 row in set (0.01 sec)
  8. mysql>

组合聚集函数

  1. - 这些聚集函数也可以集中在一个查询中使用
  1. mysql> SELECT COUNT(*) AS 成绩记录总数, MAX(score) AS 最高成绩, MIN(score) AS 最低成绩, AVG(score) AS 平均成绩 FROM student_score;
  2. +--------------------+--------------+--------------+--------------+
  3. | 成绩记录总数 | 最高成绩 | 最低成绩 | 平均成绩 |
  4. +--------------------+--------------+--------------+--------------+
  5. | 8 | 100 | 46 | 73.1250 |
  6. +--------------------+--------------+--------------+--------------+
  7. 1 row in set (0.00 sec)
  8. mysql>

隐式类型转换

  1. - 只要某个值的类型与上下文要求的类型不符,MySQL就会根据上下文环境中需要的类型对该值进行类型转换,由于这些类型转换都是MySQL自动完成的,所以也可以被称为隐式类型转换
  2. 1. 把操作数类型转换为适合操作符计算的相应类型。
  3. 1. 将函数参数转换为该函数期望的类型
  4. 1. 存储数据时,把某个值转换为某个列需要的类型(单向的 数字 --> 字符串)
  5. - MySQL会尽量把值转换为表达式中需要的类型,而不是产生错误。

4.分组查询

分组查询: GROUP BY

  1. - 针对某个列,将该列的值的相同记录分到一个组中
  1. mysql> SELECT subject, AVG(score) FROM student_score GROUP BY subject;
  2. +-----------------------------+------------+
  3. | subject | AVG(score) |
  4. +-----------------------------+------------+
  5. | 母猪的产后护理 | 73.0000 |
  6. | 论萨达姆的战争准备 | 73.2500 |
  7. +-----------------------------+------------+
  8. 2 rows in set (0.01 sec)
  9. mysql>
  1. - 对于查询结果中设定某列, 其: 既非聚合函数, 又不是分组列. 则分组后 组的该列数据是不能确定取哪个组成员的值的. 就会出现报错:
  1. mysql> SELECT number, subject, AVG(score) FROM student_score GROUP BY subject;
  2. ERROR 1055 (42000): Expression #1 of SELECT list is not in GROUP BY clause and contains nonaggregated column 'xiaohaizi.student_score.number' which is not functionally dependent on columns in GROUP BY clause; this is incompatible with sql_mode=only_full_group_by
  3. mysql>

带有WHERE子句的分组查询

  1. - 这个过程可以分成两个步骤理解: WHERE GROUP BY
  2. 1. 将记录进行过滤后分组。
  3. 1. 分别对各个分组进行数据统计, 产生最终结果

分组后的过滤条件 HAVING

  1. - 流程: WHERE => GROUP BY => HAVING
  1. SELECT subject, AVG(score) FROM student_score GROUP BY subject having subject = '母猪的产后护理';

嵌套分组

  1. - ![image.png](https://cdn.nlark.com/yuque/0/2021/png/12524106/1632580051133-ae15c9fb-a848-4606-9d61-1b63f6849af8.png#clientId=uc6b33205-bc18-4&from=paste&height=215&id=ub839d829&margin=%5Bobject%20Object%5D&name=image.png&originHeight=429&originWidth=733&originalType=binary&ratio=1&size=174877&status=done&style=none&taskId=u4f74b1ca-7e84-45b6-bc8e-91541323ff6&width=366.5)
  2. - 现在有了2个大分组,4个小分组,我们把这种对大的分组下继续分组的的情形叫做嵌套分组,如果你乐意,你可以继续把小分组划分成更小的分组。我们只需要在GROUP BY子句中把各个分组列依次写上,用逗号,分隔开就好了。比如这样:
  1. mysql> SELECT department, major, COUNT(*) FROM student_info GROUP BY department, major;
  2. +-----------------+--------------------------+----------+
  3. | department | major | COUNT(*) |
  4. +-----------------+--------------------------+----------+
  5. | 航天学院 | 电子信息 | 1 |
  6. | 航天学院 | 飞行器设计 | 1 |
  7. | 计算机学院 | 计算机科学与工程 | 2 |
  8. | 计算机学院 | 软件工程 | 2 |
  9. +-----------------+--------------------------+----------+
  10. 4 rows in set (0.00 sec)
  11. mysql>
  1. - 可以看到,在嵌套分组中,聚集函数将作用在最后一个分组列上,在这个例子中就是major列。

使用分组注意事项

  1. 1. 如果分组列中含有NULL值,那么NULL也会作为一个独立的分组存在。
  2. 1. 如果存在多个分组列,也就是嵌套分组,聚集函数将作用在最后的那个分组列上。
  3. 1. 如果查询语句中存在WHERE子句和ORDER BY子句,那么GROUP BY子句必须出现在WHERE子句之后,ORDER BY子句之前。
  4. 1. 非分组列不能单独出现在检索列表中(可以被放到聚集函数中)。
  5. 1. GROUP BY子句后也可以跟随表达式(但不能是聚集函数)。
  6. 1. WHERE子句和HAVING子句的区别。WHERE子句在分组前进行过滤,作用于每一条记录,WHERE子句过滤掉的记录将不包括在分组中。而HAVING子句在数据分组后进行过滤,作用于整个分组。

5.简单查询语句中各子句的顺序

  1. SELECT [DISTINCT] 查询列表
  2. [FROM 表名]
  3. [WHERE 布尔表达式]
  4. [GROUP BY 分组列表 ]
  5. [HAVING 分组过滤条件]
  6. [ORDER BY 排序列表]
  7. [LIMIT 开始行, 限制条数]
  1. - 其中中括号[]中的内容表示可以省略,我们在书写查询语句的时候各个子句必须严格遵守这个顺序,不然会报错的!


6.子查询

标量子查询

  1. - 第二条查询语句的搜索条件用到了第一条查询语句的查询结果。为了书写简便,我们可以把这两条语句合并到一条语句中.
  2. - 我们把第二条查询语句用小括号()扩起来作为一个操作数放到了第一条的搜索条件处,这样就起到了合并两条查询语句的作用。小括号中的查询语句也被称为子查询或者内层查询,使用内层查询的结果作为搜索条件的操作数的查询称为外层查询。如果你在一个查询语句中需要用到更多的表的话,那么在一个子查询中可以继续嵌套另一个子查询,在执行查询语句时,将按照从内到外的顺序依次执行这些查询。
  1. mysql> SELECT * FROM student_score WHERE number = (SELECT number FROM student_info WHERE name = '杜琦燕');
  2. +----------+-----------------------------+-------+
  3. | number | subject | score |
  4. +----------+-----------------------------+-------+
  5. | 20180102 | 母猪的产后护理 | 100 |
  6. | 20180102 | 论萨达姆的战争准备 | 98 |
  7. +----------+-----------------------------+-------+
  8. 2 rows in set (0.01 sec)
  9. mysql>
  1. - 在这个例子中的子查询的结果只有一个值(也就是'杜琦燕'的学号),这种子查询称之为标量子查询。正因为标量子查询单纯的代表一个值,所以它可以作为表达式的操作数来参与运算,它除了用在外层查询的搜索条件中以外,也可以被放到查询列表处,比如这样:
  1. mysql> SELECT (SELECT number FROM student_info WHERE name = '杜琦燕') AS 学号;
  2. +----------+
  3. | 学号 |
  4. +----------+
  5. | 20180102 |
  6. +----------+
  7. 1 row in set (0.00 sec)
  8. mysql>

列子查询

  1. - 很显然第一条查询语句的结果集中并不是一个单独的值,而是一个列(本例中第一条查询语句的结果集中该列包含2个值,分别是:2018010120180102),所以它对应的子查询也被称之为列子查询。因为列子查询得到的结果是多个值,相当于一个列表。我们前边的章节中说过,INNOT IN操作符正好是用来匹配列表的,上边使用的例子是使用IN操作符和子查询的结果组成表达式来作为外层查询的搜索条件的。
  1. mysql> SELECT * FROM student_score WHERE number IN (SELECT number FROM student_info WHERE major = '计算机科学与工程');
  2. +----------+-----------------------------+-------+
  3. | number | subject | score |
  4. +----------+-----------------------------+-------+
  5. | 20180101 | 母猪的产后护理 | 78 |
  6. | 20180101 | 论萨达姆的战争准备 | 88 |
  7. | 20180102 | 母猪的产后护理 | 100 |
  8. | 20180102 | 论萨达姆的战争准备 | 98 |
  9. +----------+-----------------------------+-------+
  10. 4 rows in set (0.00 sec)
  11. mysql>

行子查询

  1. - 只要子查询的结果集中最多只包含一条记录,而且这条记录中有超过一个列的数据(如果该条记录只包含一个列的话,该子查询就成了标量子查询),那么这个子查询就可以被称之为行子查询
  1. mysql> SELECT * FROM student_score WHERE (number, subject) = (SELECT number, '母猪的产后护理' FROM student_info LIMIT 1);
  2. +----------+-----------------------+-------+
  3. | number | subject | score |
  4. +----------+-----------------------+-------+
  5. | 20180104 | 母猪的产后护理 | 55 |
  6. +----------+-----------------------+-------+
  7. 1 row in set (0.01 sec)
  8. mysql>

在想要得到标量子查询或者行子查询,但又不能保证子查询的结果集只有一条记录时,应该使用LIMIT 1子句来限制记录数量。

表子查询

  1. - 如果子查询结果集中包含多行多列,那么这个子查询也可以被称之为表子查询
  1. mysql> SELECT * FROM student_score WHERE (number, subject) IN (SELECT number, '母猪的产后护理' FROM student_info WHERE major = '计算机科学与工程');
  2. +----------+-----------------------+-------+
  3. | number | subject | score |
  4. +----------+-----------------------+-------+
  5. | 20180101 | 母猪的产后护理 | 78 |
  6. | 20180102 | 母猪的产后护理 | 100 |
  7. +----------+-----------------------+-------+
  8. 2 rows in set (0.00 sec)
  9. mysql>

EXISTS和NOT EXISTS子查询

  1. - 有时候外层查询并不关心子查询中的结果是什么,而只关心子查询的结果集是不是为空集,这时可以用到这两个操作符:
操作符 示例 描述
EXISTS EXISTS (SELECT …) 当子查询结果集存在(不是空集)时表达式为真
NOT EXISTS NOT EXISTS (SELECT …) 当子查询结果集不存在(空集)时表达式为真
  1. mysql> SELECT * FROM student_score WHERE EXISTS (SELECT * FROM student_info WHERE number = 20180108);
  2. Empty set (0.00 sec)
  3. mysql>
  1. - 其中子查询的意思是在student_info表中查找学号为20180108的学生信息,很显然并没有学号为20180108的学生,所以子查询的结果集是一个空集,于是EXISTS表达式的结果为FALSE,所以外层查询也就不查了,直接返回了一个Empty set,表示没有结果。


7.连接查询

简介

  1. - 连接的本质就是把各个表中的记录都取出来依次匹配的组合加入结果集并返回给用户。
  2. - ![](https://cdn.nlark.com/yuque/0/2021/webp/12524106/1632733789853-1b450cfd-0f0f-49e4-9c80-d436ac995603.webp#clientId=uf2253c52-e6ea-4&from=paste&id=u7f35215a&margin=%5Bobject%20Object%5D&originHeight=431&originWidth=1077&originalType=url&ratio=1&status=done&style=none&taskId=u2654e492-33ce-4a17-b834-1aef37924e5)

内连接

  1. mysql> SELECT * FROM t1;
  2. +------+------+
  3. | m1 | n1 |
  4. +------+------+
  5. | 1 | a |
  6. | 2 | b |
  7. | 3 | c |
  8. +------+------+
  9. 3 rows in set (0.00 sec)
  10. mysql> SELECT * FROM t2;
  11. +------+------+
  12. | m2 | n2 |
  13. +------+------+
  14. | 2 | b |
  15. | 3 | c |
  16. | 4 | d |
  17. +------+------+
  18. 3 rows in set (0.00 sec)
  19. mysql> SELECT * FROM t1, t2;
  20. +------+------+------+------+
  21. | m1 | n1 | m2 | n2 |
  22. +------+------+------+------+
  23. | 1 | a | 2 | b |
  24. | 2 | b | 2 | b |
  25. | 3 | c | 2 | b |
  26. | 1 | a | 3 | c |
  27. | 2 | b | 3 | c |
  28. | 3 | c | 3 | c |
  29. | 1 | a | 4 | d |
  30. | 2 | b | 4 | d |
  31. | 3 | c | 4 | d |
  32. +------+------+------+------+
  33. 9 rows in set (0.00 sec)
  1. - 查询列表处的*代表从FROM语句后列出的表中选取每个列,上边的查询语句其实和下边这几种写法都是等价的:
  1. SELECT t1.m1, t1.n1, t2.m2, t2.n2 FROM t1, t2;
  2. SELECT m1, n1, m2, n2 FROM t1, t2;
  3. SELECT t1.*, t2.* FROM t1, t2;
  1. - 内连接的等价写法:
  1. SELECT * FROM t1, t2;
  2. SELECT * FROM t1 JOIN t2;
  3. SELECT * FROM t1 INNER JOIN t2;
  4. SELECT * FROM t1 CROSS JOIN t2;

连接过程

  1. - 如果我们乐意,我们可以连接任意数量张表,但是如果没有任何限制条件的话,这些表连接起来产生的笛卡尔积可能是非常巨大的。比方说3100行记录的表连接起来产生的笛卡尔积就有100×100×100=1000000行数据!所以在连接的时候过滤掉特定记录组合是有必要的
  2. - 对于内连接的两个表,驱动表中的记录在被驱动表中找不到匹配的记录,该记录不会加入到最后的结果集,我们上边提到的连接都是所谓的内连接。

外连接

  1. - MySQL中,根据选取驱动表的不同,外连接细分为2种:
  2. - 左外连接: LEFT JOIN
  3. - 选取左侧的表为驱动表。
  1. SELECT * FROM t1 LEFT [OUTER] JOIN t2 ON 连接条件 [WHERE 普通过滤条件];
  1. - 右外连接: RIGHT JOIN
  2. - 选取右侧的表为驱动表。
  1. SELECT * FROM t1 RIGHT [OUTER] JOIN t2 ON 连接条件 [WHERE 普通过滤条件];

连接查询中两种过滤条件

  1. - WHERE
  2. - WHERE子句中的过滤条件就是我们平时见的那种,不论是内连接还是外连接,凡是不符合WHERE子句中的过滤条件的记录都不会被加入最后的结果集。
  3. - ON
  4. - 对于外连接的驱动表的记录来说,如果无法在被驱动表中找到匹配ON子句中的过滤条件的记录,那么该记录仍然会被加入到结果集中,对应的被驱动表记录的各个字段使用NULL值填充。
  5. - ON子句是专门为外连接驱动表中的记录在被驱动表找不到匹配记录时应不应该把该记录加入结果集这个场景下提出的,所以如果把ON子句放到内连接中,MySQL会把它和WHERE子句一样对待,也就是说:内连接中的WHERE子句和ON子句是等价的。
  6. - 一般情况下,我们都把只涉及单表的过滤条件放到WHERE子句中,把涉及两表的过滤条件都放到ON子句中,我们也一般把放到ON子句中的过滤条件也称之为连接条件。

多表连接

  1. - 类比双表

自连接

  1. - 上边说的都是多个不同的表之间的连接,其实同一个表也可以进行连接。比方说我们可以对两个t1表来生成笛卡尔积
  1. mysql> SELECT * FROM t1, t1;
  2. ERROR 1066 (42000): Not unique table/alias: 't1'
  3. mysql>
  1. - 报错: 因为设计MySQL不允许FROM子句中出现相同的表名。我们这里需要的是两张一模一样的t1表进行连接,为了把两个一样的表区分一下,需要为表定义别名。比如这样:
  1. mysql> SELECT * FROM t1 AS table1, t1 AS table2;
  2. +------+------+------+------+
  3. | m1 | n1 | m1 | n1 |
  4. +------+------+------+------+
  5. | 1 | a | 1 | a |
  6. | 2 | b | 1 | a |
  7. | 3 | c | 1 | a |
  8. | 1 | a | 2 | b |
  9. | 2 | b | 2 | b |
  10. | 3 | c | 2 | b |
  11. | 1 | a | 3 | c |
  12. | 2 | b | 3 | c |
  13. | 3 | c | 3 | c |
  14. +------+------+------+------+
  15. 9 rows in set (0.00 sec)
  16. mysql>
  1. - 这里相当于我们为t1表定义了两个副本,一个是table1,另一个是table2,其连接过程可认为两个不同的表

表别名

  1. - 我们也可以为表来定义别名,格式与定义列的别名一致,都是用空白字符或者AS隔开,这个在表名特别长的情况下可以让语句表达更清晰一些

8.组合查询 UNION

  1. - 把两个查询语句的结果集合并到一个大的结果集中
  1. mysql> SELECT m1, n1 FROM t1 WHERE m1 < 2;
  2. +------+------+
  3. | m1 | n1 |
  4. +------+------+
  5. | 1 | a |
  6. +------+------+
  7. 1 row in set (0.00 sec)
  8. mysql> SELECT m2, n2 FROM t2 WHERE m2 > 2;
  9. +------+------+
  10. | m2 | n2 |
  11. +------+------+
  12. | 3 | c |
  13. | 4 | d |
  14. +------+------+
  15. 2 rows in set (0.00 sec)
  1. - 第一个查询是从t1表中查询m1, n1这两个列的数据,第二个查询是从t2表中查询m2, n2这两个列的数据。我们可以使用UNION直接将这两个查询语句拼接到一起:
  1. mysql> SELECT m1, n1 FROM t1 WHERE m1 < 2 UNION SELECT m2, n2 FROM t2 WHERE m2 > 2;
  2. +------+------+
  3. | m1 | n1 |
  4. +------+------+
  5. | 1 | a |
  6. | 3 | c |
  7. | 4 | d |
  8. +------+------+
  9. 3 rows in set (0.01 sec)
  10. mysql>
  1. - 第一个查询的查询列表处的m1和第二个查询的查询列表的m2对应,第一个查询的查询列表处的n1和第二个查询的查询列表的n2对应,m1/m2虽然类型不同,但MySQL会帮助我们自动进行必要的类型转换。
  2. - 这几个查询语句的结果集都可以被合并到一个大的结果集中,但是这个大的结果集总是要有展示一下列名的吧,所以就规定组合查询的结果集中显示的列名将以第一个查询中的列名为准,上边的例子就采用了第一个查询中的m1, n1作为结果集的列名。

包含或去除重复的行

  1. - UNION来合并多个查询的记录会默认过滤掉重复的记录。由于t1表和t2表都有(2, b)、(3, c)这两条记录,所以合并后的结果集就把他俩去重了:
  1. mysql> SELECT m1, n1 FROM t1;
  2. +------+------+
  3. | m1 | n1 |
  4. +------+------+
  5. | 1 | a |
  6. | 2 | b |
  7. | 3 | c |
  8. +------+------+
  9. 3 rows in set (0.00 sec)
  10. mysql> SELECT m2, n2 FROM t2;
  11. +------+------+
  12. | m2 | n2 |
  13. +------+------+
  14. | 2 | b |
  15. | 3 | c |
  16. | 4 | d |
  17. +------+------+
  18. 3 rows in set (0.00 sec)
  19. mysql> SELECT m1, n1 FROM t1 UNION SELECT m2, n2 FROM t2;
  20. +------+------+
  21. | m1 | n1 |
  22. +------+------+
  23. | 1 | a |
  24. | 2 | b |
  25. | 3 | c |
  26. | 4 | d |
  27. +------+------+
  28. 4 rows in set (0.00 sec)
  29. mysql>
  1. - 如果我们想要保留重复记录,可以使用UNION ALL来连接多个查询:
  1. mysql> SELECT m1, n1 FROM t1 UNION ALL SELECT m2, n2 FROM t2;
  2. +------+------+
  3. | m1 | n1 |
  4. +------+------+
  5. | 1 | a |
  6. | 2 | b |
  7. | 3 | c |
  8. | 2 | b |
  9. | 3 | c |
  10. | 4 | d |
  11. +------+------+
  12. 6 rows in set (0.00 sec)
  13. mysql>

组合查询中的ORDER BY和LIMIT子句

  1. - 组合查询会把各个查询的结果汇总到一块,如果我们相对最终的结果集进行排序或者只保留几行的话,可以在组合查询的语句末尾加上ORDER BYLIMIT子句
  1. mysql> SELECT m1, n1 FROM t1 UNION SELECT m2, n2 FROM t2 ORDER BY m1 DESC LIMIT 2;
  2. +------+------+
  3. | m1 | n1 |
  4. +------+------+
  5. | 4 | d |
  6. | 3 | c |
  7. +------+------+
  8. 2 rows in set (0.00 sec)
  9. #加上括号()区分查询块
  10. mysql> (SELECT m1, n1 FROM t1) UNION (SELECT m2, n2 FROM t2) ORDER BY m1 DESC LIMIT 2;
  11. +------+------+
  12. | m1 | n1 |
  13. +------+------+
  14. | 4 | d |
  15. | 3 | c |
  16. +------+------+
  17. 2 rows in set (0.01 sec)
  18. mysql>
  1. - 由于最后的结果集展示的列名是第一个查询中给定的列名,所以ORDER BY子句中指定的排序列也必须是第一个查询中给定的列名(别名也可以)
  2. - 对各小查询部分进行 group by / limit:
  1. mysql> (SELECT m1, n1 FROM t1 ORDER BY m1 DESC LIMIT 1) UNION (SELECT m2, n2 FROM t2 ORDER BY m2 DESC LIMIT 1);
  2. +------+------+
  3. | m1 | n1 |
  4. +------+------+
  5. | 3 | c |
  6. | 4 | d |
  7. +------+------+
  8. 2 rows in set (0.00 sec)
  9. mysql>
  1. - 最终结果集中的(3, 'c')其实就是查询(SELECT m1, n1 FROM t1 ORDER BY m1 DESC LIMIT 1)的结果,(4, 'd')其实就是查询(SELECT m2, n2 FROM t2 ORDER BY m2 DESC LIMIT 1)的结果。

9.数据的插入、删除和更新

插入完整的记录

  1. - 在插入完整的一条记录时,需要我们指定要插入表的名称和该条记录中全部列的具体数据,完整的语法是这样:
  1. INSERT INTO 表名 VALUES(列1的值,列2的值, ..., n的值);
  1. - VALUES语句中必须给出表中所有列的值,缺一个都不行,如果我们不知道向某个列填什么值,可以使用填入NULL(前提是该列没有声明NOT NULL属性)

插入记录的一部分

  1. - 在插入记录的时候,某些列的值可以被省略,但是这个列必须满足下边列出的某个条件之一:
  2. - 该列允许存储NULL
  3. - 该列有DEFAULT属性,给出了默认值
  1. INSERT INTO 表名 (列1,列3,列5,...) VALUES(列1的值,列3的值,列5的值, ...);
  1. - INSERT语句中指定的列顺序可以改变,但是一定要和VALUES列表中的值一一对应起来。

批量插入记录

  1. - 直接在VALUES后多加几组值,每组值用小括号()扩起来,各个组之间用逗号分隔
  1. mysql> INSERT INTO first_table(first_column, second_column) VALUES(7, 'ggg'), (8, 'hhh');
  2. Query OK, 2 rows affected (0.00 sec)
  3. Records: 2 Duplicates: 0 Warnings: 0
  4. mysql>

将某个查询的结果集插入表中

  1. INSERT INTO 目标表名(目标列1, 目标列2) SELECT 查找列1, 查找列2 FROM 查找的表名 ;
  1. - 在将某个查询的结果集插入到表中时需要注意,INSERT语句指定的列要和查询列表中的表达式一一对应

INSERT IGNORE

  1. - 于一些是主键或者具有UNIQUE约束的列或者列组合来说,它们不允许重复值的出现,比如我们把first_tablefirst_column列添加一个UNIQUE约束. 测试中: 因为first_column列有了UNIQUE约束,所以如果待插入记录的first_column列值与已有的值重复的话就会报错
  1. mysql> ALTER TABLE first_table MODIFY COLUMN first_column INT UNIQUE;
  2. Query OK, 0 rows affected (0.03 sec)
  3. Records: 0 Duplicates: 0 Warnings: 0
  4. mysql>
  1. - 对于那些是主键或者具有UNIQUE约束的列或者列组合来说,如果表中已存在的记录中没有与待插入记录在这些列或者列组合上重复的值,那么就把待插入记录插到表中,否则忽略此次插入操作
  1. mysql> INSERT IGNORE INTO first_table(first_column, second_column) VALUES(1, '哇哈哈') ;
  2. Query OK, 0 rows affected, 1 warning (0.00 sec)
  3. mysql>
  1. - 我们只是简单的在INSERT后边加了个IGNORE单词便不再报错了!对于批量插入的情况,INSERT IGNORE同样适用

INSERT ON DUPLICATE KEY UPDATE

  1. - 对于主键或者有唯一性约束的列或列组合来说,新插入的记录如果和表中已存在的记录重复的话,我们可以选择的策略不仅仅是忽略该条记录的插入,也可以选择更新这条重复的旧记录: 对于那些是主键或者具有UNIQUE约束的列或者列组合来说,如果表中已存在的记录中没有与待插入记录在这些列或者列组合上重复的值,那么就把待插入记录插到表中,否则按照规定去更新那条重复的记录中某些列的值
  1. INSERT (通用的插入语句) ON DUPLICATE KEY UPDATE ()对于重复值的默认更新处理的操作)
  1. mysql> INSERT INTO first_table (first_column, second_column) VALUES(1, '哇哈哈') ON DUPLICATE KEY UPDATE second_column = '雪碧';
  2. Query OK, 2 rows affected (0.00 sec)
  3. mysql>
  1. - 这个语句的意思就是,对于要插入的数据(1, '哇哈哈')来说,如果first_table表中已经存在first_column的列值为1的记录(因为first_column列具有UNIQUE约束),那么就把该记录的second_column列更新为'雪碧'

删除(行)数据

  1. - 普通的 where筛选 删除行数据
  1. DELETE FROM 表名 [WHERE 表达式];
  1. - 以使用LIMIT子句来限制想要删除掉的记录数量,使用ORDER BY子句来指定符合条件的记录的删除顺序
  1. DELETE FROM 表名 ORDER BY 排序根据的列名 DESC LIMIT 1;

更新数据

  1. - 我们在UPDATE单词后边指定要更新的表,然后把你想更新的列的名称和该列更新后的值写到SET单词后边,如果想更新多个列的话,它们之间用逗号,分隔开。如果我们不指定WHERE子句,那么表中所有的记录都会被更新,否则的话只有符合WHERE子句中的条件的记录才可以被更新。
  1. UPDATE 表名 SET 1=值1, 2=值2, ..., n=值n [WHERE 布尔表达式];