逻辑树分析法,对复杂问题进行拆解
create table student(
sid varchar(6) not null primary key comment '学号',
sname varchar(10) comment '学生姓名',
sage datetime comment '出生日期',
ssex varchar(10) comment '性别'
)engine=InnoDB default charset=utf8;
alter table student comment '学生';
insert into student(sid,sname,sage,ssex) values ('01','赵风','1990-01-01','男');
insert into student(sid,sname,sage,ssex) values ('02','钱雨','1990-12-21','男');
insert into student(sid,sname,sage,ssex) values ('03','孙雷','1990-05-20','男');
insert into student(sid,sname,sage,ssex) values ('04','李电','1990-08-06','男');
insert into student(sid,sname,sage,ssex) values ('05','周梅','1991-12-01','女');
insert into student(sid,sname,sage,ssex) values ('06','吴兰','1992-03-01','女');
insert into student(sid,sname,sage,ssex) values ('07','郑竹','1989-07-01','女');
insert into student(sid,sname,sage,ssex) values ('08','王菊','1990-01-20','女');
create table course(
cid varchar(10) not null primary key comment '课程编号',
cname varchar(10) comment '课程名称',
tid varchar(10) comment '老师编号'
)engine=InnoDB default charset=utf8;
alter table course comment '课程';
insert into course(cid,cname,tid) values ('01','语文','02');
insert into course(cid,cname,tid) values ('02','数学','01');
insert into course(cid,cname,tid) values ('03','英语','03');
create table teacher(
tid varchar(10) not null primary key comment '老师编号',
tname varchar(10) comment '老师姓名'
)engine=InnoDB default charset=utf8;
alter table teacher comment '老师';
insert into teacher(tid,tname) values ('01','张三');
insert into teacher(tid,tname) values ('02','李四');
insert into teacher(tid,tname) values ('03','王五');
create table sc(
sid varchar(10) comment '学号',
cid varchar(10) comment '课程编号',
score decimal(18,1) comment '分数',
primary key(sid,cid)
)engine=InnoDB default charset=utf8;
alter table sc comment '成绩';
insert into sc(sid,cid,score) values ('01','01',80);
insert into sc(sid,cid,score) values ('01','02',90);
insert into sc(sid,cid,score) values ('01','03',99);
insert into sc(sid,cid,score) values ('02','01',70);
insert into sc(sid,cid,score) values ('02','02',60);
insert into sc(sid,cid,score) values ('02','03',80);
insert into sc(sid,cid,score) values ('03','01',80);
insert into sc(sid,cid,score) values ('03','02',80);
insert into sc(sid,cid,score) values ('03','03',80);
insert into sc(sid,cid,score) values ('04','01',50);
insert into sc(sid,cid,score) values ('04','02',30);
insert into sc(sid,cid,score) values ('04','03',20);
insert into sc(sid,cid,score) values ('05','01',76);
insert into sc(sid,cid,score) values ('05','02',87);
insert into sc(sid,cid,score) values ('06','01',31);
insert into sc(sid,cid,score) values ('06','03',34);
insert into sc(sid,cid,score) values ('07','02',89);
insert into sc(sid,cid,score) values ('07','03',98);
1、查询“01”课程比“02”课程成绩高的学生的信息及课程分数
# 学生信息student(sid)、 sc(sid,cid)表
# 纵表转横表 case...when...; 横表转纵表 union all
select s.*,a.score_01,a.score_02 from
(select
sid,
sum(case when cid='01' then score end) as score_01,
sum(case when cid='02' then score end) as score_02
from sc
group by sid) a
left join student s on a.sid=s.sid
where (a.score_01 is not null or a.score_02 is not null) and a.score_01>a.score_02;
2、查询平均成绩大于等于60分的同学的学生id、学生姓名和平均成绩
# 先找出参加全部考试的学生id,和他们的平均成绩,然后左关联student表,筛选出平均成绩大于等于60即可
select a.sid,b.sname,a.avg_score from
(select sid,avg(score) as avg_score from sc group by sid having count(cid)=3) A
left join student b on a.sid=b.sid
where a.avg_score>=60;
3、查询在sc表存在成绩的学生信息
select * from student a where exists (select 1 from sc b where a.sid=b.sid);
4、查询所有同学的学生id、学生姓名、选课总数、所有课程的总成绩(没成绩的显示为null)
# student(sid,sname) sc(count(cid),sum(score))
select a.sid,a.sname,ifnull(b.count_course,0) as count_courses,b.sum_score
from student a left join (
select sid,count(cid) as count_course,sum(score) as sum_score from sc group by sid
) b on a.sid=b.sid;
5、查询有成绩的学生信息
select * from student s where s.sid in (
select distinct sid from sc
);
6、查询“李”姓老师的数量
select count(*) from teacher where tname like '李%';
7、查询学过“张三”老师授课的同学的信息
# teacher(tid,tname) course(cid,tid) sc(sid,cid) student(sid)
select * from student s where sid in (
select distinct sid from sc where cid in (
select cid from course where tid in (
select tid from teacher where tname='张三'
)
)
);
8、查询没有学全所有课程的同学的信息
# 总的课程数量<>每个学生选课的数量
select b.* from (
select sid from sc group by sid having count(cid)<>(select count(*) from course)
) a left join student b on a.sid=b.sid;
9、查询和“01”号同学学习的课程完全相同的其他同学的信息
1)找出学号=01的学生选修的所有课程
2)完全相同
select * from student where sid in
(select sid from sc where sid not in
(select sid from sc where cid not in (select cid from sc where sid='01'))
group by sid
having count(*)=(select count(*) from sc where sid='01') and sid != '01');
10、查询至少有一门课与学号为“01”的同学所学相同的同学的信息
select s.* from (
select distinct b.sid from sc b where b.cid in (select a.cid from sc a where a.sid='01')
) c left join student s on c.sid=s.sid
where c.sid<>'01';
11、查询没有学过“张三”老师讲授的任一门课程的学生姓名
select sname from student where sid not in(
select distinct sid from sc where cid in(
select cid from course where tid=(
select tid from teacher where tname='张三')
)
);
12、查询两门及其以上不及格课程的同学的id,姓名及其平均成绩
select b.*,a.avg_score from (
select sid,avg(score) as avg_score from sc where score<60 group by sid
) a left join student b on a.sid=b.sid;
13、检索“01”课程分数小于60,按分数降序排列的学生信息
select a.sid,b.sname,a.score
from sc a left join student b on a.sid=b.sid
where a.cid='01' and a.score<60
order by a.score desc;
14、按平均成绩从高到低显示所有学生的所有课程的成绩以及平均成绩
# order by 平均成绩 desc
# sid cid score cid score cid score avg_scores
# 方法一:纵表转横表 用avg函数求平均值,但是会忽略null,结果不符合逻辑
select sid,
max(case when cid='01' then score end) as sc_01,
max(case when cid='02' then score end) as sc_02,
max(case when cid='03' then score end) as sc_03,
avg(score) as avg_score
from sc
group by sid
order by avg(score) desc;
# 方法二:纵表转横表,使用sum函数,处理null值,除以课程总数,求得均值
select sid,
max(case when cid='01' then score end) as sc_01,
max(case when cid='02' then score end) as sc_02,
max(case when cid='03' then score end) as sc_03,
sum(score)/3 as avg_score
from sc
group by sid
order by sum(score)/3 desc;
15、查询各科成绩最高分、最低分和平均分,要求输出课程号和选修人数
# max(score) min(score) avg(score) count(*) group by cid
select cid,
max(score) as max_score,
min(score) as min_score,
avg(score) as avg_score,
count(*) as total
from sc
group by cid;
16、按平均成绩进行排序,显示总排名和各科排名,score重复时保留名次空缺
# rank() 横表转纵表 总排名 各科排名
select A.*,B.avg_scores,B.total_ranks
from (
select
t1.sid,
max(case when t1.cid='01' then t1.score end) as cid_01,
max(case when t1.cid='01' then t1.ranks end) as ranks_01,
max(case when t1.cid='02' then t1.score end) as cid_02,
max(case when t1.cid='02' then t1.ranks end) as ranks_02,
max(case when t1.cid='03' then t1.score end) as cid_03,
max(case when t1.cid='03' then t1.ranks end) as ranks_03
from (select *,rank() over(partition by cid order by score desc) as ranks from sc) t1
group by sid
) A left join (
select s1.*,rank() over(order by s1.avg_scores desc) as total_ranks
from (select s.sid,sum(s.score)/3 as avg_scores from sc s group by s.sid) s1
) B on A.sid=B.sid
order by B.total_ranks,A.ranks_01,A.ranks_02,A.ranks_03;
-- select t.*,rank() over(partition by t.cid order by t.score desc) as ranks from sc t;
17、按平均成绩进行排序,显示总排名和各科排名,score重复时合并名次
# dense_rank() 横表转纵表
--1、先对课程成绩排名
select *,dense_rank() over(partition by cid order by score desc) as ranks from sc;
--2、在1的基础上横表转纵表,求出各科目成绩及排名
select
t1.sid,
max(case when t1.cid='01' then t1.score end) as cid_01,
max(case when t1.cid='01' then t1.ranks end) as ranks_01,
max(case when t1.cid='02' then t1.score end) as cid_02,
max(case when t1.cid='02' then t1.ranks end) as ranks_02,
max(case when t1.cid='03' then t1.score end) as cid_03,
max(case when t1.cid='03' then t1.ranks end) as ranks_03
from (select *,dense_rank() over(partition by cid order by score desc) as ranks from sc) t1
group by sid;
--3、求出平均成绩名次,并进行排名
select s1.*,dense_rank() over(order by s1.avg_scores desc) as total_ranks from (select s.sid,sum(s.score)/3 as avg_scores from sc s group by s.sid) s1;
--4、两表关联
select A.*,B.avg_scores,B.total_ranks
from (
select
t1.sid,
max(case when t1.cid='01' then t1.score end) as cid_01,
max(case when t1.cid='01' then t1.ranks end) as ranks_01,
max(case when t1.cid='02' then t1.score end) as cid_02,
max(case when t1.cid='02' then t1.ranks end) as ranks_02,
max(case when t1.cid='03' then t1.score end) as cid_03,
max(case when t1.cid='03' then t1.ranks end) as ranks_03
from (select *,dense_rank() over(partition by cid order by score desc) as ranks from sc) t1
group by sid
) A left join (
select s1.*,dense_rank() over(order by s1.avg_scores desc) as total_ranks
from (select s.sid,sum(s.score)/3 as avg_scores from sc s group by s.sid) s1
) B on A.sid=B.sid
order by B.total_ranks,A.ranks_01,A.ranks_02,A.ranks_03;
18、统计各科成绩各分数段人数:课程id,课程名称,【100-85】,【85-70】,【70-60】,【60-0】及所占百分比
# cid cname count() 转置
# cid 【100-85】【85-70】【70-60】【60-0】
select
SC.Cid,Course.Cname,
sum(case when score>=85 and score<=100 then 1 else 0 end) as '[85-100]',
concat(sum(case when score>=85 and score<=100 then 1 else 0 end)/count(*)*100,'%') as '[85-100]%',
sum(case when score>=70 and score<85 then 1 else 0 end) as '[70-85)',
concat(sum(case when score>=70 and score<85 then 1 else 0 end)/count(*)*100,'%') as '[70-85)%',
sum(case when score>=60 and score<70 then 1 else 0 end) as '[60-70)',
concat(sum(case when score>=60 and score<70 then 1 else 0 end)/count(*)*100,'%') as '[60-70)%',
sum(case when score>=0 and score<60 then 1 else 0 end) as '[0-60)',
concat(sum(case when score>=0 and score<60 then 1 else 0 end)/count(*)*100,'%') as '[0-60)%'
from SC left join Course on SC.Cid=Course.Cid
group by SC.Cid;
19、查询各科成绩前三名的记录
select sid,cid,score from (
select sid,cid,score,row_number() over(partition by cid order by score desc) as tt from sc) A
where a.tt<4;
20、查询出只选修两门课程的学生id和姓名
# count(*)=2 group by sid
select b.sid,c.sname from
(select a.sid,count(*) from sc a group by a.sid having count(*)=2) b
left join student c on b.sid=c.sid;
21、查询名字中含有“风”字的学生信息
select * from student where sname like '%风%';
22、查询1990年出生的学生信息
select * from student where year(sage)=1990;
23、成绩不重复,查询选修“李四”老师所授课程的学生中,成绩最高的学生信息及其成绩
select b.*,a.score
from sc a left join student b on a.sid=b.sid
where a.cid=(select cid from course where tid=(select tid from teacher where tname='李四'))
order by a.score desc
limit 1;
24、成绩有重复的情况下,查询选修“李四”老师所授课程的学生中,成绩最高的学生信息及其成绩
select s.*,b.score from (
select a.*,dense_rank() over(partition by a.cid order by a.score desc) as rows_num
from sc a
where a.cid=(select cid from course where tid=(select tid from teacher where tname='李四'))) b
left join student s on b.sid=s.sid
where b.rows_num=1;
25、查询各学生的年龄,只按年份来算
select *,(year(now())-year(sage)) as age from student;
26、查询各学生的年龄,按照出生日期来算,当前月日<出生年月的月日则,年龄减一
select *,
(case when date_format(sage,'%m-%d')-date_format(now(),'%m-%d')>0 then year(now())-year(sage)-1 else year(now())-year(sage) end) as age
from student;
27、查询本周过生日的学生
select * from student where week(sage)=week(now());
28、查询本月过生日的学生
select * from student where month(sage)=month(now());
29、查询至少选修了202002学生的全部课程的学生学号
1)找出学号=202002的学生选修的所有课程
2)根据1)的结果,找出所有跟202002选修了同一门课(任意一门)的数据
3)按学号进行分组,从2)的结果中找出选课数量等于202002学生的选课数量的所有学生
select a.*
from sql02_score a
where a.cno in(select b.cno from sql02_score b where b.sno='202002')
group by a.sno
having count(*)=(select count(*) from sql02_score c where c.sno='202002');