SQL练习题

本文通过逻辑树分析法介绍了一系列复杂的SQL查询问题,包括课程成绩对比、学生信息筛选、教师授课查询、课程排名统计等多个场景。这些问题涵盖了对数据库表的深入查询和分析,旨在提升SQL技能和数据库理解。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

逻辑树分析法,对复杂问题进行拆解

在这里插入图片描述

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');
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值