1 简介
之前写过一个比较单表查询比较简单的内容。在单表查询查询中,还有较为高级的部分,诸如聚合查询,排序和分组查询,熟悉的了解关系型数据库中此类操作,可以非常方便的帮助我们实现自定义的数据提取操作。单表查询的简单查询内容可以参见
2 材料
因为所有的数据库管理系统的底层均为命令行,只不过提供界面之后更加的直观和容易操作,为了基础,仍以命令行为工具,查询不同命令的输入和输出结果,以期增加对关系型数据库的理解和使用。
楼主使用的数据库为MySQL 8.0.11
3 操作记录
单表操作的主要使用语句为SELECT语句,在SELECT语句中,可以根据自己对数据的需求,使用不同的查询条件,基本语法格式如下:
SELECT [DISTINCT] *|{字段名1, 字段名2, 字段名3,……}
FROM 表名
[WHERE 条件表达式1]
[GROUP BY 字段名 [HAVING 条件表达式2]]
[ORDER BY 字段名 [ASC|DESC]]
[LIMIT [OFFSET] 记录数]
3.1 产生数据
mysql> show databases;
+--------------------+
| Database |
+--------------------+
| act |
| information_schema |
| mysql |
| performance_schema |
| sys |
+--------------------+
5 rows in set (0.00 sec)
mysql> create database test
-> ;
Query OK, 1 row affected (0.20 sec)
mysql> show databases;
+--------------------+
| Database |
+--------------------+
| act |
| information_schema |
| mysql |
| performance_schema |
| sys |
| test |
+--------------------+
6 rows in set (0.00 sec)
使用数据库test,并在数据库中产生表student
mysql> use test;
Database changed
mysql> create table student(
-> id INT(3) PRIMARY KEY AUTO_INCREMENT,
-> name varchar(20) not null,
-> grade float,
-> gender CHAR(2)
-> );
Query OK, 0 rows affected (0.51 sec)
mysql> show tables;
+----------------+
| Tables_in_test |
+----------------+
| student |
+----------------+
1 row in set (0.00 sec)
使用insert语句向表student中插入8条数据
mysql> insert into student (name, grade, gender)
-> VALUES (“songjiang”, 40, ‘男’),
-> (“wuyong”, 100, ‘男’),
-> (“qinming”, 90, ‘男’),
-> (“husanniang”, 80, ‘男’),
-> (“sunerniang”, 66, ‘男’),
-> (“wusong”, 86, ‘男’),
-> (“linchong”, 92, ‘男’),
-> (“yanqing”, 90, ‘男’);
Query OK, 8 rows affected (0.19 sec)
Records: 8 Duplicates: 0 Warnings: 0
3.2 聚合函数
实际开发中,经常需要对某些数据进行统计,例如统计某个字段的最大值,最小值,平均值等,为了实现这些功能,MySQL提供了一些函数来实现这些功能。
3.2.1 COUNT()
COUNT函数用来统计记录的条数,语法格式如下:
select COUNT(*) from 表名
统计student表中记录数
mysql> select * from student;
+----+------------+-------+--------+
| id | name | grade | gender |
+----+------------+-------+--------+
| 1 | songjiang | 40 | 男 |
| 2 | wuyong | 100 | 男 |
| 3 | qinming | 90 | 男 |
| 4 | husanniang | 80 | 男 |
| 5 | sunerniang | 66 | 男 |
| 6 | wusong | 86 | 男 |
| 7 | linchong | 92 | 男 |
| 8 | yanqing | 90 | 男 |
+----+------------+-------+--------+
8 rows in set (0.05 sec)
mysql> select count(*) from student;
+----------+
| count(*) |
+----------+
| 8 |
+----------+
1 row in set (0.05 sec)
为查询到的记录总数起个别名
mysql> select count(*) as records from student;
+---------+
| records |
+---------+
| 8 |
+---------+
1 row in set (0.00 sec)
mysql> select count(*) records from student;
+---------+
| records |
+---------+
| 8 |
+---------+
1 row in set (0.00 sec)
3.2.2 SUM()
sum是求和函数,用于求出表中某个字段所有值的总和,语法如下:
select sum(字段名) from 表名
使用该语句可以求出指定字段值的总和
求出student表中grade成绩的总和
mysql> select sum(grade) 成绩总和 from student;
+--------------+
| 成绩总和 |
+--------------+
| 644 |
+--------------+
1 row in set (0.01 sec)
3.2.3 AVG()
AVG可以用于求出某个字段所有值得平均值
select avg(字段名) from student;
mysql> select avg(grade) from student;
+------------+
| avg(grade) |
+------------+
| 80.5 |
+------------+
1 row in set (0.00 sec)
3.2.4 MAX/min
max/min函数则是用于求最大值/最小值的函数,用于求出某个字段的最大值
mysql> select max(grade) from student;
+------------+
| max(grade) |
+------------+
| /100 |
+------------+
1 row in set (0.00 sec)
mysql> select min(grade) from student;
+------------+
| min(grade) |
+------------+
| 40 |
+------------+
1 row in set (0.00 sec)
3.3 对查询结果进行排序
从表中查询出来的数据可能是无序的,或者其排列顺序不是用户预期的。为了使查询结果满足用户的要求,可以使用ORDER BY对查询结果进行排序,语法格式如下:
SELECT 字段名1,字段名2,……
FROM 表名
ORDER BY 字段名1 [ASC | DESC],字段名2 [ASC | DESC]……
查询student表中的所有记录,按照grade进行排序
mysql> select * from student order by grade;
+----+------------+-------+--------+
| id | name | grade | gender |
+----+------------+-------+--------+
| 1 | songjiang | 40 | 男 |
| 5 | sunerniang | 66 | 男 |
| 4 | husanniang | 80 | 男 |
| 6 | wusong | 86 | 男 |
| 3 | qinming | 90 | 男 |
| 8 | yanqing | 90 | 男 |
| 7 | linchong | 92 | 男 |
| 2 | wuyong | 100 | 男 |
+----+------------+-------+--------+
8 rows in set (0.00 sec)
mysql> select * from student order by grade ASC;
+----+------------+-------+--------+
| id | name | grade | gender |
+----+------------+-------+--------+
| 1 | songjiang | 40 | 男 |
| 5 | sunerniang | 66 | 男 |
| 4 | husanniang | 80 | 男 |
| 6 | wusong | 86 | 男 |
| 3 | qinming | 90 | 男 |
| 8 | yanqing | 90 | 男 |
| 7 | linchong | 92 | 男 |
| 2 | wuyong | 100 | 男 |
+----+------------+-------+--------+
8 rows in set (0.00 sec)
mysql>
从查询结果可以看到,返回的记录按照order by指定的字段进行排序,并且默认是升序的。两次查询的结果是一只的。
按照grade 逆序排列
mysql> select * from student order by grade desc;
+----+------------+-------+--------+
| id | name | grade | gender |
+----+------------+-------+--------+
| 2 | wuyong | 100 | 男 |
| 7 | linchong | 92 | 男 |
| 3 | qinming | 90 | 男 |
| 8 | yanqing | 90 | 男 |
| 6 | wusong | 86 | 男 |
| 4 | husanniang | 80 | 男 |
| 5 | sunerniang | 66 | 男 |
| 1 | songjiang | 40 | 男 |
+----+------------+-------+--------+
8 rows in set (0.00 sec)
更新student表中yanqing行性别为NULL
mysql> update student set gender=NULL where name='yanqing';
Query OK, 1 row affected (0.15 sec)
Rows matched: 1 Changed: 1 Warnings: 0
在MySQL中,可以指定按照多个字段对查询结果进行排序。例如按照gender和grade字段进行排序,按照gender字段升序排列,按照grade字段降序排列
mysql> select * from student order by gender asc, grade desc;
+----+------------+-------+--------+
| id | name | grade | gender |
+----+------------+-------+--------+
| 8 | yanqing | 90 | NULL |
| 4 | husanniang | 80 | 女 |
| 5 | sunerniang | 66 | 女 |
| 2 | wuyong | 100 | 男 |
| 7 | linchong | 92 | 男 |
| 3 | qinming | 90 | 男 |
| 6 | wusong | 86 | 男 |
| 1 | songjiang | 40 | 男 |
+----+------------+-------+--------+
8 rows in set (0.00 sec)
上述的命令使得记录按照多个字段进行排序,可以看到输出的记录先按照gender进行升序排列,相同的gender,则按照grade逆序排列。
3.4 分组查询
在对表中数据进行统计时,也可能需要按照一定的类别进行统计,比如,分别统计student表中gender字段为“男” “女“, “NULL”的学生成绩之和。在mysql中,可以使用group by按某个字段或者多个字段中的值进行分组,字段中值相同的为1组,其语法格式如下:
SELECT 字段名1,字段名2,……
FROM 表名
GROUP BY 字段名1,字段名2,……[HAVING 条件表达式];
HAVING关键字指定条件表达式对分组后的内容进行过滤。需要注意的是,Group BY一般和聚合函数一起使用,如果查询的字段出现在Group By后,却没有包含在聚合函数中,该字段显示的是分组后第一条记录的值,这样可能会导致查询的结果与预期不符合。
3.4.1 单独使用Group By进行分组
mysql> select * from student group by gender;
ERROR 1055 (42000): Expression #1 of SELECT list is not in GROUP BY clause and contains nonaggregated column 'test.stude
nt.id' which is not functionally dependent on columns in GROUP BY clause; this is incompatible with sql_mode=only_full_g
roup_by
mysql> select * from student group by id;
+----+------------+-------+--------+
| id | name | grade | gender |
+----+------------+-------+--------+
| 1 | songjiang | 40 | 男 |
| 2 | wuyong | 100 | 男 |
| 3 | qinming | 90 | 男 |
| 4 | husanniang | 80 | 女 |
| 5 | sunerniang | 66 | 女 |
| 6 | wusong | 86 | 男 |
| 7 | linchong | 92 | 男 |
| 8 | yanqing | 90 | NULL |
+----+------------+-------+--------+
8 rows in set (0.00 sec)
mysql> select count(*), gender from student group by gender;
+----------+--------+
| count(*) | gender |
+----------+--------+
| 5 | 男 |
| 2 | 女 |
| 1 | NULL |
+----------+--------+
3 rows in set (0.00 sec)
3.4.2 Group By和聚合函数一起使用
Group BY和聚合函数一起使用,可以统计处某个或者某些字段在一个分组中的最大值,最小值,平均值。
将student表按照gender字段值进行分组查询,计算出每个分组有多少个学生
mysql> select count(*), gender from student group by gender;
+----------+--------+
| count(*) | gender |
+----------+--------+
| 5 | 男 |
| 2 | 女 |
| 1 | NULL |
+----------+--------+
3 rows in set (0.00 sec)
从查询结果可以看出,group by对student表按照gender字段中不同的值进行了分组,并通过count函数统计出每个分组的个数。
3.4.3 Group By和Having关键字一起使用
Having关键字与where字句作用相同,都用于设置条件表达式对查询结果进行过滤,两者的区别在于,Having关键字后可以跟聚合函数,而where子句不可以。通常情况先Having关键字都和Group By一起使用,用于对分组后的结果进行过滤。
将student表中按照gender字段进行分组排序,查询出grade字段值之和小于300的分组
mysql> select sum(grade), gender from student group by gender having sum(grade)<300;
+------------+--------+
| sum(grade) | gender |
+------------+--------+
| 146 | 女 |
| 90 | NULL |
+------------+--------+
rows in set (0.06 sec)
对gender值为’男’的所有学生其grade字段值之和进行查询
mysql> select sum(grade), gender from student where gender='男';
+------------+--------+
| sum(grade) | gender |
+------------+--------+
| 408 | 男 |
+------------+--------+
1 row in set (0.00 sec )
3.5 使用LIMIT限制查询结果的数量
查询数据时,可能会返回很多数据,而用户需要的记录可能只是其中的一条或者几条,比如实现分页功能,每页显示十条数据,每次查询就只需要十条数据。为此,MySQL提供了LIMIT关键字,可以指定查询结果从那一条记录开始,以及一共查询多少条信息,语法格式如下:
SELECT 字段名1,字段名2,……
FROM 表名
LIMIT [OFFSET,] 记录数
LIMIT:后面可以跟2个参数,“OFFSET”:为可选值,表示偏移量,如果偏移量为0则从查询结果的第一条记录开始…以此类推,如果不指定其默认值为0。
“记录数”:表示返回查询记录的条数。
查询表student的前四条数据
mysql> select * from student limit 4
-> ;
+----+------------+-------+--------+
| id | name | grade | gender |
+----+------------+-------+--------+
| 1 | songjiang | 40 | 男 |
| 2 | wuyong | 100 | 男 |
| 3 | qinming | 90 | 男 |
| 4 | husanniang | 80 | 女 |
+----+------------+-------+--------+
4 rows in set (0.00 sec)
查询student表中grade字段值从第5到第8位的学生,从高到底
mysql> select * from student order by grade desc limit 4, 4;
+----+------------+-------+--------+
| id | name | grade | gender |
+----+------------+-------+--------+
| 6 | wusong | 86 | 男 |
| 4 | husanniang | 80 | 女 |
| 5 | sunerniang | 66 | 女 |
| 1 | songjiang | 40 | 男 |
+----+------------+-------+--------+
rows in set (0.00 sec)
注意:LIMIT后跟了两个参数,两个参数以逗号隔开,第一个参数表示偏移量为4,即从第五条记录开始查询,第二个参数表示一共返回4条数据,即从第五到第八位。是哟个ORDER BY DESC使学生记录按照grade字段从高到低的顺序进行排列。
显示所有student记录,以grade排序,逆序
mysql> select * from student order by grade desc;
+----+------------+-------+--------+
| id | name | grade | gender |
+----+------------+-------+--------+
| 2 | wuyong | 100 | 男 |
| 7 | linchong | 92 | 男 |
| 3 | qinming | 90 | 男 |
| 8 | yanqing | 90 | NULL |
| 6 | wusong | 86 | 男 |
| 4 | husanniang | 80 | 女 |
| 5 | sunerniang | 66 | 女 |
| 1 | songjiang | 40 | 男 |
+----+------------+-------+--------+
8 rows in set (0.00 sec)
3.6 函数(列表)
MySQL中提供了丰富的函数,通过这些函数可以简化用户对数据的操
作,MySQL中的函数包括数学函数,字符串函数,日期和时间函数条件判断函数,加密函数,由于函数较多,不一一讲解。
3.6.1 数学函数
函数名称 | 作用 |
ABS(x) | 绝对值 |
SQRT(x) | 返回x的非负二次方根 |
MOD(x, y) | 返回x被y除后的余数 |
FLOOR(x) | 返回不大于x的最大整数 |
ROUND(x, y) | 对x进行四舍五入整数,小数点保持y位 |
TRUNCATE(x, y) | 舍去x中小数点y位后面的数 |
SIGN(x) | 返回x的符号,-1, 0, 1 |
3.6.2 字符串函数
函数 | 作用 |
length(str) | 返回str的长度 |
concat(s1, s2,…) | 返回一个或者多个字符串连接产生的新字符串 |
trim(str) | 删除字符串两侧的空格 |
replace(str, s1, s2) | 使用字符串s2替换字符串str中所有的字符串s1 |
substring(str, n, len) | 返回字符串str的子串,起始位置为n,长度为len |
reverse(str) | 返回字符串反转后的结果 |
locate(s1, str) | 返回字符串s1在字符串str中的起始位置。 |
3.6.3 日期和时间函数
函数 | 作用 |
curdate() | 获取当前日期 |
curtime() | 获取当前时间 |
sysdate() | 获取当前系统日期和时间 |
time_to_sec() | 返回将时间转换成秒的结果 |
adddate() | 执行日期的加运算 |
subdate() | 执行日期的减运算 |
date_format() | 格式化输出日期和时间值 |
3.6.4 条件判断 函数
函数 | 作用 |
IF(expr, v1, v2) | 如果表达式expr为true,返回v1,否则v2 |
IFNULL(v1, v2) | 如果v1不为NULL,返回v1,否则返回v2 |
CASE expr when v1 then r1 [when v2 then r2…] else rn end | 如果expr的值等于v1,v2等值,则返回对应位置的then之后的结果,否则返回else之后的结果 |
3.6.5 加密函数
函数 | 作用 |
MD5(str) | 对字符串str进行MD5加密 |
Encode(str, pwd_str) | 使用pwd作为密码加密字符串str |
decode(str, pwd_str) | 使用pwd作为密码解密字符串str |
查询student表中所有记录,将各个字段值使用下划线”_”连接起来
mysql> select concat(id, "_", name, "_", grade, "_", gender) from student;
+------------------------------------------------+
| concat(id, "_", name, "_", grade, "_", gender) |
+------------------------------------------------+
| 1_songjiang_40_男 |
| 2_wuyong_100_男 |
| 3_qinming_90_男 |
| 4_husanniang_80_女 |
| 5_sunerniang_66_女 |
| 6_wusong_86_男 |
| 7_linchong_92_男 |
| NULL |
+------------------------------------------------+
8 rows in set (0.00 sec)
查询student表中的id和gender值,如果gender为男,则返回1否则返回0
mysql> select id, if(gender='男', 1, 0) from student;
+----+------------------------+
| id | if(gender='男', 1, 0) |
+----+------------------------+
| 1 | 1 |
| 2 | 1 |
| 3 | 1 |
| 4 | 0 |
| 5 | 0 |
| 6 | 1 |
| 7 | 1 |
| 8 | 0 |
+----+------------------------+
8 rows in set (0.00 sec)
4 为表和字段取别名
4.1 为表取别名
在查询数据时,可以为表和字段取别名,这个别名可以代替指定的表和字段。如果表名很长,用起来不方便,则可以用一个别名来代替,语法如下:
select * from 表名 [as] 别名
as关键字可以省略。
如下例子所示:
mysql> select * from student as s where s.gender='女';
+----+------------+-------+--------+
| id | name | grade | gender |
+----+------------+-------+--------+
| 4 | husanniang | 80 | 女 |
| 5 | sunerniang | 66 | 女 |
+----+------------+-------+--------+
2 rows in set (0.00 sec)
mysql> select * from student s where s.gender='女';
+----+------------+-------+--------+
| id | name | grade | gender |
+----+------------+-------+--------+
| 4 | husanniang | 80 | 女 |
| 5 | sunerniang | 66 | 女 |
+----+------------+-------+--------+
2 rows in set (0.00 sec)
4.2 为字段取别名
SELECT 字段名 [AS] 别名[,字段名 [AS] 别名,……] FROM 表名;
如例子:
mysql> select name as stu_name, gender stu_gender from student;
+------------+------------+
| stu_name | stu_gender |
+------------+------------+
| songjiang | 男 |
| wuyong | 男 |
| qinming | 男 |
| husanniang | 女 |
| sunerniang | 女 |
| wusong | 男 |
| linchong | 男 |
| yanqing | NULL |
+------------+------------+
8 rows in set (0.00 sec)
5 总结
本文主要是简要讲述了Mysql中单表的高级查询,通过熟练掌握命令行操作,可以加深对mysql的基本使用,对于关系型数据库的理解非常有助。在Java spring对数据库的增删改查操作时,使用的是同样的原理。