子查询又叫嵌套查询。它通常可以位于SELECT后面 FROM后面
WHERE后面,共三种使用场景。当我们查询一个表没有办法实现功能的时候,就需要使用子查询
上面我们讲到了分组查询,可以查询每个老师所带学生的最低分,
但是我们刚才查出来之后,我们只能看到teacher_id,但是我们并不知道teacher_id对应的是那个老师,这个时候我们最好是显示老师的名字是比较好的,可以用子查询实现
场景一 : select后面
语法 :
select 字段名,(查询语句) from 表名;
如 : 查询所有学生的信息并显示老师的名字
select *,(
select name from
teacher where id=teacher_id
) as teacher_name from student ;
如 :
查询每个老师的学生的 最大分数,最小分数,平均分数,分数总和,学生人数,老师名字
select max(score),min(score),sum(score),avg(score),count(*),(
select name from
teacher where id=teacher_id
) as teacher_name from
student group by teacher_id ;
注意 :
当位于SELECT后面时,要注意
1.一定要在两个表之间找好对应关系(teacher.id必须是主键或者必须保证teacher.id在teacher表中是唯一的)
2.子查询中只能有一个字段(子查询的结果必须是一行一列)
使用子查询的时候,建议大家养成使用别名的好习惯,这样可以让我们的查询语句更加清晰。别名可以用来命令新字段,也可以用来命名新表.
场景二 : from后面
还是学生表student,我们要将成绩进行分级,并且显示汉字的分级与字母的分级。这里可以使用子查询。相当于给student“新增”了2个字段
如 : 使用子查询 对成绩划分等级, score<60 ,评级C 并且是差,score>=60 且 score<80 评级B并且是良,score>=80
评级是A并且是优
select *,
case rank
when ‘A’ then ‘优’
when ‘B’ then ‘良’
when ‘C’ then ‘差’
end rank_ch
from (
select *,
case
when score < 60
then ‘C’
when score >=60
and score <80 then ‘B’
when score >=80
then ‘A’
end as rank
from student
) a;
注意 :
当位于FROM后面时,要注意
1.我们可以把子查询当成一张表
2.必须要有别名,因为子查询优先被执行,子查询的别名,可以让别的查询当做表或者列去操作
场景三 : where后面
如 : 在不知道teacher_id 和 老师名字的对应关系的情况下,想查询出张老师下面的所有学生信息
select * from student where teacher_id in (
select id from teacher where name=‘张老师’
);
注意 :
当位于WHERE后面时,要注意
多条数据要用in而不要用=,如果确定子查询的结果为一行一列的话,就可以用 = 等于
如果返回结果为多行一列的话 要用 in , 一列是必须的,必须是一列
3.子查询中的SELECT后面只能有一个字段(多个字段的话会报错)
合并查询,合并查询的结果
Union 会去除重复项
Union all 不会去除重复项
如 : 查询出 teacher_id = 1 的所有学生信息
select * from student where teacher_id=1;
如 : 查询出 学生分数大于60的所有学生信息
select * from student where score > 60;
如 : 查询出 学生分数大于60 或
teacher_id = 1 的所有学生信息(去除重复)
// 用 or 实现
select * from student where teacher_id=1 or score > 60;
// 用 union实现
select * from student where teacher_id=1
union
select * from student where score > 60;
如 : 查询出 学生分数大于60 或
teacher_id = 1 的所有学生信息(可重复)
select * from student where teacher_id=1
union all
select * from student where score > 60;
总结和注意 :
union / union all
它俩的作用是把两张表或者更多表合并成一张表
前者会去重(去重的依据是,UNION时SELECT出来的字段如果对应相等则认为是同一条记录,这的逻辑我们可以参考Java equals)
后者则不会去重,它会保留两张表中的所有记录,但是它性能高(因为去重操作要花时间),
尽量使用union all,把去重这个工作交给代码去完成,这样可以减少MYSQL服务器的压力
使用union / union all的时候要注意:
1.参与合并的表,它们SELECT出来的字段数量必须一致(强制规则)
2.参与合并的表,它们SELECT出来的字段的类型建议一一对应(非强制,但是最好遵循这条规则)
3.参与合并的表,它们SELECT出来的字段的顺序建议一致(非强制,但是最好遵循这条规则)
演示示例:子查询
select version() ;显示当前MySQL软件的版本
select database();显示当前所处数据库是哪个
select
char_length(‘中国’);返回字符个数。
select
length(‘中国’);返回字符所占字节数,MySQL中,一个UTF8编码的汉字占3个字节
select
concat( ‘a’, ‘b’,
‘c’, ‘d’);返回 ‘abcd’。字符串拼接函数
select
concat_ws( ‘=’, ‘a’,
‘b’, ‘c’);返回
‘a=b=c’。字符串拼接函数,第一个是拼接间隔符
select
upper(‘abcd’);返回ABCD。将参数中所有小写字母转换为大写
select
lower(‘ABCD’);返回abcd。将参数中所有大写字母转换为小写
select
substring( ‘系统信息类’,
1, 3 );返回 系统信。第2个参数代表从1开始的第几个字符,第3个参数代表截取字符个数
select
trim(’ abc ');返回 abc。用于删去参数左右的所有空格
select
curdate();返回当前日期
select
curtime();返回当前时间
select
now();返回当前日期时间
select
unix_timestamp();返回当前日期时间对应的时间戳(单位秒)
select
unix_timestamp(‘2018-05-24 20:00:00’);返回参数指定的日期时间对应的时间戳(单位秒)
select
from_unixtime(1527163397);返回参数指定时间戳(单位秒)对应的日期时间
select
datediff( ‘2018-05-23’, now()
);返回两个参数对应日期相差的天数(用第一个参数减第二个参数)
select
adddate( now(), -2 );返回指定天数前/后的日期时间(第一个参数是日期时间,第二个参数是天数,向后加是正数,向前减是负数)
select year(‘2019-02-24’);返回2019 获得年份
select month(‘2019-02-24’) 返回2 获得月份
select day(‘2019-02-24’) 返回 24 获取日
select
if( <判断条件>,
<条件为真时的返回值>, <条件为假时的返回值> );相当于Java中的三目运算符<判断条件>
? <条件为真的返回值> : <条件为假的返回值>。
如select
if(1=1, 2, 3);返回2。
select
ifnull(<表达式或者字段>, <表达式或者字段为NULL时的返回值>);通常用于给有可能有NULL的情况下的提供默认值。
select ifnull(null,‘无名氏’) ; null这里可以写列名
就会把该列值为null的 以无名氏显示
select ifnull(name,‘无名氏’) from teacher ;