数据库点滴之T-SQL面试语句,练练手

1. 用一条SQL语句 查询出每门课都大于80分的学生姓名

  • name kecheng fenshu
  • 张三   语文       81
  • 张三   数学       75
  • 李四   语文       76
  • 李四   数学       90
  • 王五   语文       81
  • 王五   数学     100
  • 王五   英语       90

思路:这里不能直接用 分数>80这样的比较条件来查询的到结果,因为要求没门成绩都大于80。我们可以反过来思考,如果有一门成绩小于80,那么就不符合要求。先找出成绩表中成绩<80的多有学生姓名,不能重复,然后再用not in找出不再这个集合中的学生姓名。

  1. create table #成绩(姓名varchar(20),课程名称varchar(20),分数int
  2. insert into #成绩values 
  3. ('张三',     '语文',       81), 
  4. ('张三',     '数学',       75), 
  5. ('李四',     '语文',       76), 
  6. ('李四',     '数学',       90), 
  7. ('王五',     '语文',      81), 
  8. ('王五',     '数学',       100), 
  9. ('王五',     '英语',       90) 
  10.  
  11. select distinct(姓名) from #成绩 where 姓名 not in(select distinct(姓名) from #成绩 where 分数<=80) 

经luofer提示还有一种思路,是用group by + hvaing,这绝对是一种好方法。我估计出这个题的人就是要考察这个知识,代码如下:

  1. select 姓名 from #成绩  
  2. group by 姓名 
  3. having min(分数)>80 

2. 学生表 如下:

  • 自动编号     学号       姓名 课程编号 课程名称 分数
  •        1         2005001    张三      0001          数学       69
  •        2         2005002    李四      0001          数学       89
  •        3         2005001    张三      0001          数学       69

删除除了自动编号不同,其他都相同的学生冗余信息

思路:这个和上面的一样,也不能直接删除,而是要先找出自动编号不相同,其他都相同的行,这个要使用group by语句,并且将其他的字段都放在group by后面,这样找出来的行都是没有冗余的行,然后随便保留其中一个自动编号,删除其他的行。

  1. create table #成绩(自动编号 int, 学号 int,姓名 varchar(20),课程编号 int,课程名称 varchar(20),分数 int
  2. insert into #成绩 values 
  3. (1,2005001 ,'张三',  1,   '语文',       81), 
  4. (2,2005001 ,'李四',  1,   '语文',       81), 
  5. (3,2005001 ,'张三',  1,   '语文',       81), 
  6. (4,2005001 ,'张三',  1,   '语文',       81) 
  7.  
  8. select * from #成绩 
  9. drop table #成绩 
  10.  
  11. delete from #成绩 where 自动编号 not in 
  12. (select MIN(自动编号) from #成绩 group by 学号,姓名,课程编号,课程名称,分数) 

经【广岛之恋】的提醒发现另外一种思路,代码如下:

  1. delete from #成绩 where 自动编号 not in 
  2. (select distinct(a.自动编号) from #成绩 a join #成绩 b on a.自动编号>b.自动编号  
  3. where a.学号=b.学号 and a.姓名=b.姓名 and a.课程编号=b.课程编号 and a.分数=b.分数) 

3. 一个叫department的表,里面只有一个字段name,一共有4条纪录,分别是a,b,c,d,对应四个球对,现在四个球对进行比赛,用一条sql语句显示所有可能的比赛组合。

思路:这是一个组合问题,就是说四个不同的元素有多少种不同的两两组合。现在要把这个问题用sql语句实现。既然这四个元素是不相同的,我们可以将这个表当成两个集合,求他们的笛卡尔积,然后再从笛卡尔积中找到那些元素不相同的,并且不重复的组合。

  1. create table #department(taname char(1)) 
  2. insert into #department values 
  3. ('a'),('b'),('c'),('d'
  4.  
  5. --下面两条语句都可以,多谢wanglinglong提醒 
  6. select a.taname,b.taname from #department a,#department b where a.taname < b.taname 
  7. select a.taname,b.taname from #department a,#department b where a.taname > b.taname 

4.怎么把这样一个表

  • year  month  amount
  • 1991      1         1.1
  • 1991      2         1.2
  • 1991      3         1.3
  • 1991      4         1.4
  • 1992      1         2.1
  • 1992      2         2.2
  • 1992      3         2.3
  • 1992      4         2.4

查成这样一个结果

  • year  m1 m2 m3 m4
  • 1991 1.1 1.2 1.3 1.4
  • 1992 2.1 2.2 2.3 2.4

思路:这个很明显是一个行列转换,首先会想到pivot。结果中有m1,m2,m3,m4四个新的列,他们需要从原来的行中转换。

  1. create table #sales(years int,months int,amount float
  2. insert into #sales values 
  3. (1991,   1,     1.1), 
  4. (1991,   2,     1.2), 
  5. (1991,   3,     1.3), 
  6. (1991,   4,     1.4), 
  7. (1992,   1,     2.1), 
  8. (1992,   2,     2.2), 
  9. (1992,   3,     2.3), 
  10. (1992,   4,     2.4) 
  11.  
  12. select pt.years,[1] as m1,[2] as m2,[3] as m3,[4] as m4  
  13. from (select sod.amount,sod.months,sod.years as years from  #sales sod)  so  
  14. pivot 
  15. (min(so.amount) for so.months in ([1], [2],[3],[4])) as pt 

注意[1],[2],[3],[4]中括号不可缺少,否则会出错。还有一种写法是使用子查询,这个要新建4个子查询进而得到新的列:

  1. select a.years, 
  2. (select m.amount from #sales m where months=1 and m.years=a.years) as m1, 
  3. (select m.amount from #sales m where months=2 and m.years=a.years) as m2, 
  4. (select m.amount from #sales m where months=3 and m.years=a.years) as m3, 
  5. (select m.amount from #sales m where months=4 and m.years=a.years) as m4 
  6. from #sales a group by a.years 

5.有两个表A和B,均有key和value两个字段,如果B的key在A中也有,就把B的value换为A中对应的value。这道题的SQL语句怎么写?

思路:这个问题看似简单,只要一个update语句,然后找到相同的key,更新value字段就可以了。可能你首先会写成这样:update #b set #b.value=(select #a.value from #a where #a.keys=#b.keys)。但是要注意的是如果仅仅找相同的key会有很多匹配,更新的时候会出现错误,所有要在外层限制。

  1. create table #a(keys int , value varchar(10)) 
  2. insert into #a values 
  3. (1,'aa'), 
  4. (2,'ab'), 
  5. (3,'ac'
  6. create table #b(keys int , value varchar(10)) 
  7. insert into #b values 
  8. (1,'aa'), 
  9. (2,'a'), 
  10. (3,'a'
  11.  
  12. update #b set #b.value=(select #a.value from #a where #a.keys=#b.keys) where #b.keys in 
  13. (select #b.keys from #b,#a where #a.keys=#b.keys and #a.value<>#b.value) 

在luofer的提醒之,有了第二个思路

  1. update #b set #b.value=s.value 
  2. from (select * from #a except select * from #b) s where s.keys=#b.keys 

luofer是牛人啊!

6. 两张关联表,删除主表中已经在副表中没有的信息。

思路:这个就是存在关系,可以使用in,也可以使用exists。

  1. create table #zhubiao(id int,name varchar(5)) 
  2. insert into #zhubiao values 
  3. (1,'aa'), 
  4. (2,'ab'), 
  5. (3,'ac'
  6. create table #fubiao(id int, grade varchar(5)) 
  7. insert into #fubiao values 
  8. (1,'aa'), 
  9. (2,'ab'
  10.  
  11. delete from #zhubiao where id not in(select b.id from #fubiao b) 
  12. delete from #zhubiao where not exists(select 1 from #fubiao where #zhubiao.id=#fubiao.id) 

7. 原表:

  • courseid coursename score
  • 1        java    70
  • 2        oracle    90
  • 3            xml    40
  • 4            jsp    30
  • 5      servlet    80

为了便于阅读,查询此表后的结果显式如下(及格分数为60):

  • courseid coursename score   mark
  • 1             java   70  pass
  • 2            oracle    90  pass
  • 3              xml   40  fail
  • 4             jsp    30  fail
  • 5       servlet    80     pass

思路:这个就很直接了,使用case语句判断一下。

  1. create table #scores(course int,coursename varchar(10),score int
  2. insert into #scores values 
  3. (1, 'java', 70 ), 
  4. (2, 'oracle', 90), 
  5. (3, 'xmls', 40), 
  6. (4, 'jsp', 30),  
  7. (5, 'servlet', 80 ) 
  8.  
  9. select course,coursename, 
  10. case when score>60 then 'pass' else 'fail' end as mark 
  11. from #scores 

8. 原表:

  • id proid proname
  • 1      1        M
  • 1      2         F
  • 2      1        N
  • 2      2        G
  • 3      1        B
  • 3      2        A

查询后的表:

  • id pro1 pro2
  •  1  M      F
  •  2  N      G
  •  3  B      A

思路:依旧是行列转换,这个在面试中的几率很高。这个语句还是有两种写法,如下:

  1. create table #table1(id int,proid int,proname char
  2. insert into #table1 values 
  3. (1, 1, 'M'), 
  4. (1, 2, 'F'),  
  5. (2, 1, 'N'),  
  6. (2, 2, 'G'),  
  7. (3, 1, 'B'),  
  8. (3, 2, 'A'
  9.  
  10. select id,  
  11. (select proname from #table1 where proid=1 and id=b.id) as pro1, 
  12. (select proname from #table1 where proid=2 and id=b.id) as pro2 
  13. from #table1 b group by id 
  14.  
  15. select d.id,[1] as pro1,[2] as pro2 from 
  16. (select b.id,b.proid,b.proname from #table1 b) as c 
  17. pivot 
  18. (min(c.proname) for c.proid in([1],[2])) as d 

9. 如下

  • 表a
  • 列     a1 a2
  • 记录 1   a
  •           1   b
  •           2   x
  •           2   y
  •           2   z

用select能选成以下结果吗?

1 ab

2 xyz

思路:这个开始想使用行列转换来写,没有成功,后来没有办法只好用游标,代码如下:

  1. create table #table2(id int , value varchar(10)) 
  2. insert into #table2 values 
  3. (1,'a'), 
  4. (1,'b'), 
  5. (2,'x'), 
  6. (2,'y'), 
  7. (2,'z'
  8. create table #table3(id int,value varchar(100) );insert into #table3(id,value) select distinct(id),'' from #table2 
  9.  
  10. declare @id int,@name varchar(10) 
  11. declare mycursor cursor for select * from #table2 
  12. open mycursor 
  13. fetch next from mycursor into @id,@name 
  14. while (@@Fetch_Status = 0) 
  15. begin 
  16. update #table3 set value=value+@name where id=@id 
  17. fetch next from mycursor into @id,@name  
  18. end 
  19. close mycursor 
  20. deallocate mycursor 
  21.  
  22. select * from #table3 

有两个要注意的地方,

a.#table3里面的value字段初始值如果不设置的话默认是null,后面更新的时候null+’a’任然是null,***得到的value永远是null。所以默认是”

b.第二个fetch语句一定要放在begin和end之间,要不然会死循环的,不常用的语句写起来很不爽快

原文链接:http://www.cnblogs.com/tylerdonet/archive/2011/10/07/2200500.html

【编辑推荐】

  1. 数据库点滴之SQL级联删除
  2. 数据库点滴之精妙SQL语句
  3. SQL点滴之性能优化其实没有那么神秘
  4. SQL点滴之几个有点偏的语句
  5. SQL点滴之如何编辑数据

文章来源网络,作者:运维,如若转载,请注明出处:https://shuyeidc.com/wp/231312.html<

(0)
运维的头像运维
上一篇2025-04-19 15:37
下一篇 2025-04-19 15:38

相关推荐

  • 个人主题怎么制作?

    制作个人主题是一个将个人风格、兴趣或专业领域转化为视觉化或结构化内容的过程,无论是用于个人博客、作品集、社交媒体账号还是品牌形象,核心都是围绕“个人特色”展开,以下从定位、内容规划、视觉设计、技术实现四个维度,详细拆解制作个人主题的完整流程,明确主题定位:找到个人特色的核心主题定位是所有工作的起点,需要先回答……

    2025-11-20
    0
  • 社群营销管理关键是什么?

    社群营销的核心在于通过建立有温度、有价值、有归属感的社群,实现用户留存、转化和品牌传播,其管理需贯穿“目标定位-内容运营-用户互动-数据驱动-风险控制”全流程,以下从五个维度展开详细说明:明确社群定位与目标社群管理的首要任务是精准定位,需明确社群的核心价值(如行业交流、产品使用指导、兴趣分享等)、目标用户画像……

    2025-11-20
    0
  • 香港公司网站备案需要什么材料?

    香港公司进行网站备案是一个涉及多部门协调、流程相对严谨的过程,尤其需兼顾中国内地与香港两地的监管要求,由于香港公司注册地与中国内地不同,其网站若主要服务内地用户或使用内地服务器,需根据服务器位置、网站内容性质等,选择对应的备案路径(如工信部ICP备案或公安备案),以下从备案主体资格、流程步骤、材料准备、注意事项……

    2025-11-20
    0
  • 如何企业上云推广

    企业上云已成为数字化转型的核心战略,但推广过程中需结合行业特性、企业痛点与市场需求,构建系统性、多维度的推广体系,以下从市场定位、策略设计、执行落地及效果优化四个维度,详细拆解企业上云推广的实践路径,精准定位:明确目标企业与核心价值企业上云并非“一刀切”的方案,需先锁定目标客户群体,提炼差异化价值主张,客户分层……

    2025-11-20
    0
  • PS设计搜索框的实用技巧有哪些?

    在PS中设计一个美观且功能性的搜索框需要结合创意构思、视觉设计和用户体验考量,以下从设计思路、制作步骤、细节优化及交互预览等方面详细说明,帮助打造符合需求的搜索框,设计前的规划明确使用场景:根据网站或APP的整体风格确定搜索框的调性,例如极简风适合细线条和纯色,科技感适合渐变和发光效果,电商类则可能需要突出搜索……

    2025-11-20
    0

发表回复

您的邮箱地址不会被公开。必填项已用 * 标注