跳至主要內容

50道SQL经典面试题(下)

Mr.暴走の海鸽约 2090 字大约 7 分钟

50道SQL经典面试题(下)

原创 丶平凡世界 SQL数据库开发open in new window

最近在收集SQL每日一题时,找到这套比较经典的SQL面试题。

我根据题目重新梳理了一遍,包括表结构,表之间的关系,测试数据,题目,参考答案等。其中大部分参考答案在各种数据库平台上通用。

由于题目数量较多(足足50道题),小伙伴们可能不容易消化理解,于是将内容分为上下两篇,希望对你有所帮助。

一、表结构

1、学生表

Student(Sid,Sname,Sage,Ssex)
学生编号,学生姓名,出生年月,学生性别

2、课程表

Course(Cid,Cname,Tid) 
课程编号,课程名称,教师编号

3、教师表

Teacher(Tid,Tname)
教师编号,教师姓名

4、成绩表

SC(Sid,Cid,Score)
学生编号,课程编号,分数

二、表之间的关系

四张表之间的关系如下图:

我们来解读一下上面的关系:

1、课程表Course的课程编号(Cid)作为主键,在成绩表(SC)中可以看到一个或多个学生的课程分数,两表之间是属于1:n的关系。同理学生表(Student)与成绩表(SC)也是1:n的关系

2、教师表Teacher的教师编号(Tid)作为主键,在课程表(Course)中可以带一门或多门课程,两表之间也是属于1:n的关系。

三、测试数据

1、学生表

--建表语句
CREATE TABLE Student (
  SID VARCHAR (10),
  Sname nvarchar (10),
  Sage datetime,
  Ssex nvarchar (10)
)

--插入测试数据
INSERT INTO Student VALUES('01' , N'赵雷' , '1990-01-01' , N'男')
INSERT INTO Student VALUES('02' , N'钱电' , '1990-12-21' , N'男')
INSERT INTO Student VALUES('03' , N'孙风' , '1990-05-20' , N'男')
INSERT INTO Student VALUES('04' , N'李云' , '1990-08-06' , N'男')
INSERT INTO Student VALUES('05' , N'周梅' , '1991-12-01' , N'女')
INSERT INTO Student VALUES('06' , N'吴兰' , '1992-03-01' , N'女')
INSERT INTO Student VALUES('07' , N'郑竹' , '1989-07-01' , N'女')
INSERT INTO Student VALUES('08' , N'王菊' , '1990-01-20' , N'女')

结果如下:

2、课程表

--建表语句
CREATE TABLE Course (
  CID VARCHAR (10),
  Cname nvarchar (10),
  TID VARCHAR (10)
)
--插入测试数据
INSERT INTO Course VALUES('01' , N'语文' , '02')
INSERT INTO Course VALUES('02' , N'数学' , '01')
INSERT INTO Course VALUES('03' , N'英语' , '03')

结果如下:

3、教师表

--建表语句
CREATE TABLE Teacher (
  TID VARCHAR (10),
  Tname nvarchar (10)
)
--插入测试数据
INSERT INTO Teacher VALUES('01' , N'张三')
INSERT INTO Teacher VALUES('02' , N'李四')
INSERT INTO Teacher VALUES('03' , N'王五')

结果如下:

4、成绩表

--建表语句
CREATE TABLE SC (
  SID VARCHAR (10),
  CID VARCHAR (10),
  score DECIMAL (18, 1)
)
--插入测试数据
INSERT INTO SC VALUES('01' , '01' , 80)
INSERT INTO SC VALUES('01' , '02' , 90)
INSERT INTO SC VALUES('01' , '03' , 99)
INSERT INTO SC VALUES('02' , '01' , 70)
INSERT INTO SC VALUES('02' , '02' , 60)
INSERT INTO SC VALUES('02' , '03' , 80)
INSERT INTO SC VALUES('03' , '01' , 80)
INSERT INTO SC VALUES('03' , '02' , 80)
INSERT INTO SC VALUES('03' , '03' , 80)
INSERT INTO SC VALUES('04' , '01' , 50)
INSERT INTO SC VALUES('04' , '02' , 30)
INSERT INTO SC VALUES('04' , '03' , 20)
INSERT INTO SC VALUES('05' , '01' , 76)
INSERT INTO SC VALUES('05' , '02' , 87)
INSERT INTO SC VALUES('06' , '01' , 31)
INSERT INTO SC VALUES('06' , '03' , 34)
INSERT INTO SC VALUES('07' , '02' , 89)
INSERT INTO SC VALUES('07' , '03' , 98)

结果如下:

四、面试题及参考答案

26、查询出只选修两门课程的学生学号和姓名

SELECT SID,Sname
FROM Student
WHERE SID in
(SELECT SID FROM
  (SELECT SID,COUNT(CID) 课程数
   FROM SC GROUP BY SID 
  ) A
 WHERE A.课程数=2
)

27、查询男生、女生人数

SELECT Ssex,COUNT(Ssex) 人数
FROM Student
GROUP BY Ssex

28、查询名字中含有「风」字的学生信息

SELECT * FROM Student
WHERE Sname like '%风%'

29、查询同名同性学生名单,并统计这些人数

SELECT A.*,B.人数
FROM Student A
LEFT JOIN 
(SELECT Sname,Ssex,COUNT(*) 人数
FROM Student GROUP BY Sname,Ssex
) B
ON A.Sname=B.Sname and A.Ssex=B.Ssex
WHERE B.人数>1

30、查询 1990 年出生的学生名单

SELECT * FROM Student
WHERE YEAR(Sage)=1990

31、查询每门课程的平均成绩,结果按平均成绩降序排列,平均成绩相同时,按课程编号升序排列

SELECT CID,AVG(score) 平均成绩
FROM SC
GROUP BY CID ORDER BY 平均成绩 DESC,CID

32、查询平均成绩大于等于 85 的所有学生的学号、姓名和平均成绩

SELECT A.SID,A.Sname,B.平均成绩
FROM Student A
LEFT JOIN 
(SELECT SID,AVG(score) 平均成绩
FROM SC GROUP BY SID
) B on A.SID=B.SID
WHERE B.平均成绩>85

33、查询课程名称为「数学」,且分数低于 60 的学生姓名和分数

SELECT B.Sname,A.score FROM
(SELECT * FROM SC
WHERE score<60 
and CID=
(SELECT CID FROM Course
WHERE Cname='数学'
)
) A
LEFT JOIN Student B on A.SID=B.SID

34、查询所有学生的课程及分数情况(存在学生没成绩,没选课的情况)

SELECT A.SID,B.CID,B.score
FROM Student A
LEFT JOIN SC B on A.SID=B.SID

35、查询任何一门课程成绩在 70 分以上的姓名、课程名称和分数

SELECT A.Sname,D.Cname,D.score
FROM 
(SELECT B.*,C.Cname
FROM
(SELECT * FROM SC WHERE score>70) B
LEFT JOIN Course C on B.CID=C.CID
) D
LEFT JOIN Student A on D.SID=A.SID

36、查询不及格的课程学生姓名,课程名及分数

SELECT C.Sname,B.Cname,A.score FROM SC A
JOIN Course B ON A.CID=B.CID
JOIN Student C ON A.SID=C.SID
WHERE A.score<60

37、查询课程编号为01且课程成绩在80分以上的学生的学号和姓名

SELECT A.SID,B.Sname
FROM 
(SELECT * FROM SC
WHERE score>80 and CID='01'
) A
LEFT JOIN Student B on A.SID=B.SID

38、求每门课程的学生人数(假设每个学生都有参加考试且有成绩)

SELECT CID,COUNT(*) 学生人数
FROM SC
GROUP BY CID

39、成绩不重复,查询选修「张三」老师所授课程的学生中,成绩最高的学生信息及其成绩

SELECT TOP 1 * FROM SC
WHERE CID=
(SELECT CID
FROM Course
WHERE TID=
(SELECT TID FROM Teacher
WHERE Tname='张三'
)
)
ORDER BY score DESC

40、成绩有重复的情况下,查询选修「张三」老师所授课程的学生中,成绩最高的学生信息及其成绩

SELECT * FROM 
( SELECT *,DENSE_RANK()OVER(ORDER BY score DESC) A
FROM SC
WHERE CID=
(
SELECT CID FROM Course
WHERE TID=
(SELECT TID FROM Teacher
WHERE Tname='张三'
)
)
)B
WHERE B.A=1

41、查询每门功成绩最好的前两名

SELECT * FROM
(SELECT *,ROW_NUMBER()OVER (PARTITION BY CID ORDER BY score DESC)A
FROM SC
)B
WHERE B.A<3

42、统计每门课程的学生选修人数(超过5人的课程才统计),要求输出课程号和选修人数,查询结果按人数降序排列,若人数相同,按课程号升序排列

SELECT CID,COUNT(SID) 选修人数
FROM SC
GROUP BY CID
HAVING COUNT(SID)>5
ORDER BY 选修人数 DESC,CID

43、检索至少选修两门课程的学生学号

SELECT SID FROM SC
GROUP BY SID
HAVING COUNT(CID)>=2

44、查询选修了全部课程的学生信息

SELECT SID FROM SC
GROUP BY SID 
HAVING  COUNT(CID)=
(SELECT DISTINCT COUNT(1) a
FROM Course)

45、查询各学生的年龄,只按年份来算

SELECT SID,DATEDIFF(Year,Sage,GETDATE()) 年龄
FROM Student

46、按照出生日期来算,当前月日小于出生日期的月日则年龄减1岁

SELECT *,
(CASE WHEN  
CONVERT(INT,'1'+SUBSTRING(CONVERT(VARCHAR(10),Sage,112),5,8))
 < CONVERT(int,'1'+SUBSTRING(CONVERT(VARCHAR(10),GETDATE(),112),5,8))
THEN DATEDIFF(YY,Sage,GETDATE())
ELSE DATEDIFF(YY,Sage,GETDATE())-1 
END
)age
FROM Student

47、查询本周过生日的学生

SELECT *,(
CASE WHEN DATENAME(wk,CONVERT
(DATETIME,
(CONVERT(VARCHAR(10),YEAR(GETDATE()))
+SUBSTRING(CONVERT(VARCHAR(10),Sage,112),5,8)
)
)
)=DATENAME(WK,GETDATE())
THEN 1 ELSE 0 END
) 生日提醒
FROM Student

48、查询下周过生日的学生

SELECT *,(
CASE WHEN DATENAME(wk,CONVERT
(DATETIME,
(CONVERT(VARCHAR(10),YEAR(GETDATE()))
+SUBSTRING(CONVERT(VARCHAR(10),Sage,112),5,8)
)
)
)=DATENAME(WK,GETDATE())+1
THEN 1 ELSE 0 END
) 生日提醒
FROM Student

49、查询本月过生日的学生

SELECT *,(
CASE WHEN MONTH(
CONVERT(DATETIME,
(CONVERT(VARCHAR(10),YEAR(GETDATE()))
+SUBSTRING(CONVERT(VARCHAR(10),Sage,112),5,8)
)
)
)=MONTH(GETDATE())
THEN 1 ELSE 0 end) 生日提醒
FROM Student

50、查询下月过生日的学生

SELECT *,(
CASE WHEN MONTH(
CONVERT(DATETIME,
(CONVERT(VARCHAR(10),YEAR(GETDATE()))
+SUBSTRING(CONVERT(VARCHAR(10),Sage,112),5,8)
)
)
)=MONTH(GETDATE())+1
then 1 else 0 end)生日提醒
FROM Student

以上就是50道经典SQL面试题的全部内容