Sql Server里巧用Case将多行显示的数据合并为一行显示

本文涉及的产品
云数据库 RDS SQL Server,独享型 2核4GB
简介:

昨晚在CSDN论坛上看到有某个人问了类似这样的一个问题,现有三个数据表,分别是学生表,课程表,成绩表。它们的结构与样例数据如下:

学生表:
学生Id    姓名
1             张三
2             李四
3             王五


课程表:
课程Id    课程名
1             语文
2             化学
3             外语
4             物理


成绩表
学生Id     课程Id        成绩
1               1                 60
1               2                 70
1               3                 65
1               4                 90
2               1                 80
2               2                 65
2               3                 85
2               4                 80
3               1                 50
3               2                 75
3               3                 85
3               4                 60

现要求在一行中输出每个学生的所有课程的成绩单,显示样例格式如下:
姓名        语文        化学       外语        物理
张三        60            70           65            90
李四        80            65           85            80
王五        50            75           85            60

当大家看到这样的题目时会优先考虑到的是怎样的一条SQL语句呢?嵌套Select?对,在将行转换为列时,也许这种方法是最优先考虑到(或者你是高手,所以不是优先考虑到这个而是其它shades_smile.gif),所以我最开始也写出了下面这条语句:

None.gif SELECT  B.姓名,
None.gif(
SELECT  成绩  FROM  成绩表  INNER   JOIN  课程  ON  成绩表.课程ID = 课程.课程ID  WHERE  成绩表.学生ID = B.学生ID  AND  课程.课程名 = ' 语文 '  )  AS  语文,
None.gif(
SELECT  成绩  FROM  成绩表  INNER   JOIN  课程  ON  成绩表.课程ID = 课程.课程ID  WHERE  成绩表.学生ID = B.学生ID  AND  课程.课程名 = ' 化学 '  )  AS  化学,
None.gif(
SELECT  成绩  FROM  成绩表  INNER   JOIN  课程  ON  成绩表.课程ID = 课程.课程ID  WHERE  成绩表.学生ID = B.学生ID  AND  课程.课程名 = ' 外语 '  )  AS  外语,
None.gif(
SELECT  成绩  FROM  成绩表  INNER   JOIN  课程  ON  成绩表.课程ID = 课程.课程ID  WHERE  成绩表.学生ID = B.学生ID  AND  课程.课程名 = ' 物理 '  )  AS  物理
None.gif
FROM  学生 B
None.gif

这样我们的目的是达到了,但后来我又想了一下,因为我们要的数据其实都在 成绩表里,只不过现有的是用行来存放,那我们怎么将它转换为列显示呢?嗯,这也许就要搬出聚合函数加Case条件来处理了!最终的SQL语句如下:
None.gif SELECT  姓名,
None.gif
MAX ( CASE  课程名  WHEN   ' 语文 '   THEN  成绩  ELSE   0   END AS  语文,
None.gif
MAX ( CASE  课程名  WHEN   ' 化学 '   THEN  成绩  ELSE   0   END AS  化学,
None.gif
MAX ( CASE  课程名  WHEN   ' 外语 '   THEN  成绩  ELSE   0   END AS  外语,
None.gif
MAX ( CASE  课程名  WHEN   ' 物理 '   THEN  成绩  ELSE   0   END AS  物理
None.gif
FROM  ( SELECT  B.姓名,C.课程名,D.成绩  FROM  成绩表 D 
None.gif
INNER   JOIN  学生 B  ON  B.学生ID = D.学生ID 
None.gif
INNER   JOIN  课程 C  ON  C.课程ID = D.课程ID)  AS  TMP  GROUP   BY  姓名
None.gif

运行后,也是可以得到正确的数据,下面给出测试代码,大家可以直接在SQL查询分析器里运行

None.gif CREATE   TABLE  学生 (学生ID  INT , 姓名  VARCHAR ( 20 ))
None.gif
CREATE   TABLE  课程 (课程ID  INT , 课程名  VARCHAR ( 20 ))
None.gif
CREATE   TABLE  成绩表 (学生ID  INT , 课程ID  INT , 成绩  INT )
None.gif
None.gif
INSERT   INTO  学生
None.gif
SELECT   1 , ' 张三 '   UNION   ALL
None.gif
SELECT   2 , ' 李四 '   UNION   ALL
None.gif
SELECT   3 , ' 王五 '
None.gif
None.gif
INSERT   INTO  课程
None.gif
SELECT   1 , ' 语文 '   UNION   ALL
None.gif
SELECT   2 , ' 化学 '   UNION   ALL
None.gif
SELECT   3 , ' 外语 '   UNION   ALL
None.gif
SELECT   4 , ' 物理 '
None.gif
None.gif
INSERT   INTO  成绩表
None.gif
SELECT   1 , 1 , 60   UNION   ALL
None.gif
SELECT   1 , 2 , 70   UNION   ALL
None.gif
SELECT   1 , 3 , 65   UNION   ALL
None.gif
SELECT   1 , 4 , 90   UNION   ALL
None.gif
SELECT   2 , 1 , 80   UNION   ALL
None.gif
SELECT   2 , 2 , 65   UNION   ALL
None.gif
SELECT   2 , 3 , 85   UNION   ALL
None.gif
SELECT   2 , 4 , 80   UNION   ALL
None.gif
SELECT   3 , 1 , 50   UNION   ALL
None.gif
SELECT   3 , 2 , 75   UNION   ALL
None.gif
SELECT   3 , 3 , 85   UNION   ALL
None.gif
SELECT   3 , 4 , 60
None.gif
None.gif
-- 方法一
None.gif
SELECT  B.姓名,
None.gif(
SELECT  成绩  FROM  成绩表  INNER   JOIN  课程  ON  成绩表.课程ID = 课程.课程ID  WHERE  成绩表.学生ID = B.学生ID  AND  课程.课程名 = ' 语文 '  )  AS  语文,
None.gif(
SELECT  成绩  FROM  成绩表  INNER   JOIN  课程  ON  成绩表.课程ID = 课程.课程ID  WHERE  成绩表.学生ID = B.学生ID  AND  课程.课程名 = ' 化学 '  )  AS  化学,
None.gif(
SELECT  成绩  FROM  成绩表  INNER   JOIN  课程  ON  成绩表.课程ID = 课程.课程ID  WHERE  成绩表.学生ID = B.学生ID  AND  课程.课程名 = ' 外语 '  )  AS  外语,
None.gif(
SELECT  成绩  FROM  成绩表  INNER   JOIN  课程  ON  成绩表.课程ID = 课程.课程ID  WHERE  成绩表.学生ID = B.学生ID  AND  课程.课程名 = ' 物理 '  )  AS  物理
None.gif
FROM  学生 B
None.gif
None.gif
-- 方法二
None.gif
SELECT  姓名,
None.gif
MAX ( CASE  课程名  WHEN   ' 语文 '   THEN  成绩  ELSE   0   END AS  语文,
None.gif
MAX ( CASE  课程名  WHEN   ' 化学 '   THEN  成绩  ELSE   0   END AS  化学,
None.gif
MAX ( CASE  课程名  WHEN   ' 外语 '   THEN  成绩  ELSE   0   END AS  外语,
None.gif
MAX ( CASE  课程名  WHEN   ' 物理 '   THEN  成绩  ELSE   0   END AS  物理
None.gif
FROM  ( SELECT  B.姓名,C.课程名,D.成绩  FROM  成绩表 D 
None.gif
INNER   JOIN  学生 B  ON  B.学生ID = D.学生ID 
None.gif
INNER   JOIN  课程 C  ON  C.课程ID = D.课程ID)  AS  TMP  GROUP   BY  姓名
None.gif
None.gif
None.gif
DROP   TABLE  学生
None.gif
DROP   TABLE  课程
None.gif
DROP   TABLE  成绩表

PS:用嵌套SELECT与用聚合函数加Case两者的效率如何,我没有测试,各位有兴趣的可测试一下 shades_smile.gif
本文转自Kingthy博客园博客,原文链接:http://www.cnblogs.com/kingthy/archive/2008/04/29/1175697.html ,如需转载请自行联系原作者
相关实践学习
使用SQL语句管理索引
本次实验主要介绍如何在RDS-SQLServer数据库中,使用SQL语句管理索引。
SQL Server on Linux入门教程
SQL Server数据库一直只提供Windows下的版本。2016年微软宣布推出可运行在Linux系统下的SQL Server数据库,该版本目前还是早期预览版本。本课程主要介绍SQLServer On Linux的基本知识。 相关的阿里云产品:云数据库RDS SQL Server版 RDS SQL Server不仅拥有高可用架构和任意时间点的数据恢复功能,强力支撑各种企业应用,同时也包含了微软的License费用,减少额外支出。 了解产品详情: https://www.aliyun.com/product/rds/sqlserver
相关文章
|
6天前
|
SQL 存储 Oracle
Oracle的PL/SQL定义变量和常量:数据的稳定与灵动
【4月更文挑战第19天】在Oracle PL/SQL中,变量和常量扮演着数据存储的关键角色。变量是可变的“魔术盒”,用于存储程序运行时的动态数据,通过`DECLARE`定义,可在循环和条件判断中体现其灵活性。常量则是不可变的“固定牌”,一旦设定值便保持不变,用`CONSTANT`声明,提供程序稳定性和易维护性。通过 `%TYPE`、`NOT NULL`等特性,可以更高效地管理和控制变量与常量,提升代码质量。善用两者,能优化PL/SQL程序的结构和性能。
|
14天前
|
SQL 人工智能 算法
【SQL server】玩转SQL server数据库:第二章 关系数据库
【SQL server】玩转SQL server数据库:第二章 关系数据库
52 10
|
14天前
|
SQL 算法 数据库
【SQL server】玩转SQL server数据库:第三章 关系数据库标准语言SQL(二)数据查询
【SQL server】玩转SQL server数据库:第三章 关系数据库标准语言SQL(二)数据查询
82 6
|
2天前
|
SQL 关系型数据库 MySQL
:“You have an error in your SQL syntax; check the manual that corresponds to your MySQL server versi
:“You have an error in your SQL syntax; check the manual that corresponds to your MySQL server versi
9 0
|
6天前
|
SQL Oracle 关系型数据库
Oracle的PL/SQL游标属性:数据的“导航仪”与“仪表盘”
【4月更文挑战第19天】Oracle PL/SQL游标属性如同车辆的导航仪和仪表盘,提供丰富信息和控制。 `%FOUND`和`%NOTFOUND`指示数据读取状态,`%ROWCOUNT`记录处理行数,`%ISOPEN`显示游标状态。还有`%BULK_ROWCOUNT`和`%BULK_EXCEPTIONS`增强处理灵活性。通过实例展示了如何在数据处理中利用这些属性监控和控制流程,提高效率和准确性。掌握游标属性是提升数据处理能力的关键。
|
6天前
|
SQL Oracle 安全
Oracle的PL/SQL循环语句:数据的“旋转木马”与“无限之旅”
【4月更文挑战第19天】Oracle PL/SQL中的循环语句(LOOP、EXIT WHEN、FOR、WHILE)是处理数据的关键工具,用于批量操作、报表生成和复杂业务逻辑。LOOP提供无限循环,可通过EXIT WHEN设定退出条件;FOR循环适用于固定次数迭代,WHILE循环基于条件判断执行。有效使用循环能提高效率,但需注意避免无限循环和优化大数据处理性能。掌握循环语句,将使数据处理更加高效和便捷。
|
6天前
|
SQL Oracle 关系型数据库
Oracle的PL/SQL条件控制:数据的“红绿灯”与“分岔路”
【4月更文挑战第19天】在Oracle PL/SQL中,IF语句与CASE语句扮演着数据流程控制的关键角色。IF语句如红绿灯,依据条件决定程序执行路径;ELSE和ELSIF提供多分支逻辑。CASE语句则是分岔路,按表达式值选择执行路径。这些条件控制语句在数据验证、错误处理和业务逻辑中不可或缺,通过巧妙运用能实现高效程序逻辑,保障数据正确流转,支持企业业务发展。理解并熟练掌握这些语句的使用是成为合格数据管理员的重要一环。
|
6天前
|
SQL Oracle 关系型数据库
Oracle的PL/SQL表达式:数据的魔法公式
【4月更文挑战第19天】探索Oracle PL/SQL表达式,体验数据的魔法公式。表达式结合常量、变量、运算符和函数,用于数据运算与转换。算术运算符处理数值计算,比较运算符执行数据比较,内置函数如TO_CHAR、ROUND和SUBSTR提供多样化操作。条件表达式如CASE和NULLIF实现灵活逻辑判断。广泛应用于SQL查询和PL/SQL程序,助你驾驭数据,揭示其背后的规律与秘密,成为数据魔法师。
|
9天前
|
SQL 安全 网络安全
IDEA DataGrip连接sqlserver 提示驱动程序无法通过使用安全套接字层(SSL)加密与 SQL Server 建立安全连接的解决方法
IDEA DataGrip连接sqlserver 提示驱动程序无法通过使用安全套接字层(SSL)加密与 SQL Server 建立安全连接的解决方法
19 0
|
14天前
|
SQL 存储 数据挖掘
数据库数据恢复—RAID5上层Sql Server数据库数据恢复案例
服务器数据恢复环境: 一台安装windows server操作系统的服务器。一组由8块硬盘组建的RAID5,划分LUN供这台服务器使用。 在windows服务器内装有SqlServer数据库。存储空间LUN划分了两个逻辑分区。 服务器故障&初检: 由于未知原因,Sql Server数据库文件丢失,丢失数据涉及到3个库,表的数量有3000左右。数据库文件丢失原因还没有查清楚,也不能确定数据存储位置。 数据库文件丢失后服务器仍处于开机状态,所幸没有大量数据写入。 将raid5中所有磁盘编号后取出,经过硬件工程师检测,没有发现明显的硬件故障。以只读方式将所有磁盘进行扇区级的全盘镜像,镜像完成后将所
数据库数据恢复—RAID5上层Sql Server数据库数据恢复案例