`
happinessmoon
  • 浏览: 76044 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

sql 普通行列转换

阅读更多

/*
标题:普通行列转换(version 2.0)
作者:爱新觉罗.毓华(十八年风雨,守得冰山雪莲花开)
时间:2008-03-09
地点:广东深圳
说明:普通行列转换(version 1.0)仅针对sql server 2000提供静态和动态写法,version 2.0增加sql server 2005的有关写法。

问题:假设有张学生成绩表(tb)如下:
姓名 课程 分数
张三 语文 74
张三 数学 83
张三 物理 93
李四 语文 74
李四 数学 84
李四 物理 94
想变成(得到如下结果):
姓名 语文 数学 物理
---- ---- ---- ----
李四 74   84   94
张三 74   83   93
-------------------
*/

create table tb(姓名 varchar(10) , 课程 varchar(10) , 分数 int)
insert into tb values('张三' , '语文' , 74)
insert into tb values('张三' , '数学' , 83)
insert into tb values('张三' , '物理' , 93)
insert into tb values('李四' , '语文' , 74)
insert into tb values('李四' , '数学' , 84)
insert into tb values('李四' , '物理' , 94)
go

--SQL SERVER 2000 静态SQL,指课程只有语文、数学、物理这三门课程。(以下同)
select 姓名 as 姓名 ,
 
max(case 课程 when '语文' then 分数 else 0 end) 语文,
 
max(case 课程 when '数学' then 分数 else 0 end) 数学,
 
max(case 课程 when '物理' then 分数 else 0 end) 物理
from tb
group by 姓名

--SQL SERVER 2000 动态SQL,指课程不止语文、数学、物理这三门课程。(以下同)
declare @sql varchar(8000)
set @sql = 'select 姓名 '
select @sql = @sql + ' , max(case 课程 when ''' + 课程 + ''' then 分数 else 0 end) [' + 课程 + ']'
from (select distinct 课程 from tb) as a
set @sql = @sql + ' from tb group by 姓名'
exec(@sql)

--SQL SERVER 2005 静态SQL。
select * from (select * from tb) a pivot (max(分数) for 课程 in (语文,数学,物理)) b

--SQL SERVER 2005 动态SQL。
declare @sql varchar(8000)
select @sql = isnull(@sql + '],[' , '') + 课程 from tb group by 课程
set @sql = '[' + @sql + ']'
exec ('select * from (select * from tb) a pivot (max(分数) for 课程 in (' + @sql + ')) b')

---------------------------------

/*
问题:在上述结果的基础上加平均分,总分,得到如下结果:
姓名 语文 数学 物理 平均分 总分
---- ---- ---- ---- ------ ----
李四 74   84   94   84.00  252
张三 74   83   93   83.33  250
*/

--SQL SERVER 2000 静态SQL。
select 姓名 姓名,
 
max(case 课程 when '语文' then 分数 else 0 end) 语文,
 
max(case 课程 when '数学' then 分数 else 0 end) 数学,
 
max(case 课程 when '物理' then 分数 else 0 end) 物理,
 
cast(avg(分数*1.0) as decimal(18,2)) 平均分,
 
sum(分数) 总分
from tb
group by 姓名

--SQL SERVER 2000 动态SQL。
declare @sql varchar(8000)
set @sql = 'select 姓名 '
select @sql = @sql + ' , max(case 课程 when ''' + 课程 + ''' then 分数 else 0 end) [' + 课程 + ']'
from (select distinct 课程 from tb) as a
set @sql = @sql + ' , cast(avg(分数*1.0) as decimal(18,2)) 平均分 , sum(分数) 总分 from tb group by 姓名'
exec(@sql)

--SQL SERVER 2005 静态SQL。
select m.* , n.平均分 , n.总分 from
(
select * from (select * from tb) a pivot (max(分数) for 课程 in (语文,数学,物理)) b) m,
(
select 姓名 , cast(avg(分数*1.0) as decimal(18,2)) 平均分 , sum(分数) 总分 from tb group by 姓名) n
where m.姓名 = n.姓名

--SQL SERVER 2005 动态SQL。
declare @sql varchar(8000)
select @sql = isnull(@sql + ',' , '') + 课程 from tb group by 课程
exec ('select m.* , n.平均分 , n.总分 from
(select * from (select * from tb) a pivot (max(分数) for 课程 in (
' + @sql + ')) b) m ,
(select 姓名 , cast(avg(分数*1.0) as decimal(18,2)) 平均分 , sum(分数) 总分 from tb group by 姓名) n
where m.姓名 = n.姓名
')

drop table tb   

------------------
--
----------------

/*
问题:如果上述两表互相换一下:即表结构和数据为:
姓名 语文 数学 物理
张三 74  83  93
李四 74  84  94
想变成(得到如下结果):
姓名 课程 分数
---- ---- ----
李四 语文 74
李四 数学 84
李四 物理 94
张三 语文 74
张三 数学 83
张三 物理 93
--------------
*/

create table tb(姓名 varchar(10) , 语文 int , 数学 int , 物理 int)
insert into tb values('张三',74,83,93)
insert into tb values('李四',74,84,94)
go

--SQL SERVER 2000 静态SQL。
select * from
(
select 姓名 , 课程 = '语文' , 分数 = 语文 from tb
union all
select 姓名 , 课程 = '数学' , 分数 = 数学 from tb
union all
select 姓名 , 课程 = '物理' , 分数 = 物理 from tb
) t
order by 姓名 , case 课程 when '语文' then 1 when '数学' then 2 when '物理' then 3 end

--SQL SERVER 2000 动态SQL。
--
调用系统表动态生态。
declare @sql varchar(8000)
select @sql = isnull(@sql + ' union all ' , '' ) + ' select 姓名 , [课程] = ' + quotename(Name , '''') + ' , [分数] = ' + quotename(Name) + ' from tb'
from syscolumns
where name! = N'姓名' and ID = object_id('tb') --表名tb,不包含列名为姓名的其它列
order by colid asc
exec(@sql + ' order by 姓名 ')

--SQL SERVER 2005 动态SQL。
select 姓名 , 课程 , 分数 from tb unpivot (分数 for 课程 in([语文] , [数学] , [物理])) t

--SQL SERVER 2005 动态SQL,同SQL SERVER 2000 动态SQL。

--------------------
/*

问题:在上述的结果上加个平均分,总分,得到如下结果:
姓名 课程   分数
---- ------ ------
李四 语文   74.00
李四 数学   84.00
李四 物理   94.00
李四 平均分 84.00
李四 总分   252.00
张三 语文   74.00
张三 数学   83.00
张三 物理   93.00
张三 平均分 83.33
张三 总分   250.00
------------------
*/

select * from
(
select 姓名 as 姓名 , 课程 = '语文' , 分数 = 语文 from tb
union all
select 姓名 as 姓名 , 课程 = '数学' , 分数 = 数学 from tb
union all
select 姓名 as 姓名 , 课程 = '物理' , 分数 = 物理 from tb
union all
select 姓名 as 姓名 , 课程 = '平均分' , 分数 = cast((语文 + 数学 + 物理)*1.0/3 as decimal(18,2)) from tb
union all
select 姓名 as 姓名 , 课程 = '总分' , 分数 = 语文 + 数学 + 物理 from tb
) t
order by 姓名 , case 课程 when '语文' then 1 when '数学' then 2 when '物理' then 3 when '平均分' then 4 when '总分' then 5 end

drop table tb

分享到:
评论

相关推荐

    sql 普通行列转换

    【SQL 普通行列转换】是数据库查询中常见的需求,尤其在数据分析和报表生成时。这个话题主要涉及如何将数据表中的行数据转换为列数据,以便更方便地展示和分析。在本文中,我们将重点讨论如何在SQL Server 2000和SQL...

    SQL语句普通行列转换.docx

    ### SQL语句普通行列转换详解 #### 一、背景与目的 本文档旨在介绍如何使用SQL Server中的行列转换功能来解决常见的数据展示需求。通过具体的示例,将原始的宽表(每一行代表一个学生的所有科目成绩)转换为长表...

    sql 行专列 列转行 普通行列转换

    在SQL数据库操作中,行与列的转换是常见的数据处理需求。这通常涉及到将一列的数据拆分成多行,或者将多行数据合并为一列。这种操作在数据分析、报表生成以及数据清洗过程中非常常见。在SQL Server中,我们可以利用...

    精典的SQL语句。行转列,列转行的语句

    一、行列转换--普通 在实际应用中,我们经常需要将行转换为列,或者将列转换为行。例如,我们有一个学生成绩表(CJ),其中包含姓名、科目和成绩三列。现在,我们想将其转换为一个新的表格,其中每行代表一个学生,...

    经典SQL脚本大全

    │ │ 5.1.1 标识列与普通列互相转换的示例.sql │ │ 5.2.1 查表法按日期生成流水号的示例.sql │ │ 5.2.1 查表法生成流水号的示例.sql │ │ 5.2.2 使用编号表按日期生成流水号的示例.sql │ │ 5.2.2 使用编号表...

    Sqlserver2000经典脚本

    │ 5.1.1 标识列与普通列互相转换的示例.sql │ │ 5.2.1 查表法按日期生成流水号的示例.sql │ │ 5.2.1 查表法生成流水号的示例.sql │ │ 5.2.2 使用编号表按日期生成流水号的示例.sql │ │ ...

    SQL 通过行动态生成列

    在SQL查询中,将行数据转换为列,也称为行转列或行列互换,是数据分析和报表制作中常见的需求。这种操作对于呈现表格数据,尤其是处理统计汇总时非常有用。"SQL 通过行动态生成列"这个主题,就是探讨如何在SQL中实现...

    SQL经典短小代码

    #### 普通行列转换(version2.0) 此知识点进一步拓展了行列转换的应用场景,特别是对于SQL Server 2005及以上版本的支持。 **问题背景:** 假设有一张学生成绩表(`tb`),包含学生的姓名、所学科目及成绩。需要将其...

    sql超级经典(练习题+答案)

    --17.2 SQL 2000 动态 --17.3 有关sql 2005的动静态写法参见我的文章《普通行列转换(version 2.0)》或《普通行列转换(version 3.0)》。--18、查询各科成绩最高分、最低分和平均分:以如下形式显示:课程ID,课程name...

    表的相关查询(最全面的查询语句集合)

    ### 普通行列转换 #### SQL Server 2000 行转列 对于 SQL Server 2000,可以使用动态 SQL 来实现行转列的功能。下面的示例展示了如何将数据表中的行转换为列: ```sql CREATE TABLE tb (name VARCHAR(10), γ ...

    Oracle事例

    sql> create [unique] index index_name on table_name(column,.. asc/desc) tablespace sql> tablespace_name [pctfree integer] [initrans integer] [maxtrans integer] sql> [logging | nologging] [nosort] ...

    初试水晶报表

    实现交叉表有两种常见方法:一是通过SQL在数据库端预先组织数据,然后以普通报表形式展示;二是直接利用水晶报表的内置功能来创建交叉表。对于非标准数据,可能需要创建视图或者使用SQL命令、存储过程来预处理数据。...

    数据库选择题练习整理.pdf

    8. 完整的交叉表查询需要选择(A 行标题、列标题和值),以便呈现行列结构的数据。 9. 在E-R图中,用来表示实体的图形是(A 矩形)。 10. SQL的含义是(A 结构化查询语言),它是用于管理和处理关系数据库的标准...

    BO报表中文教程(非常好的教程)

    - **创建步骤**:详细介绍如何将普通表格转换为交叉表。 **11.11 清除单元格和删除表** - **单元格清除**:指导如何清除表格中的单元格内容。 - **表格删除**:说明如何安全地删除整个表格。 **11.12 设置表和表...

    Crystal Report报表设计培训

    此外,Crystal Reports支持多种报表样式和图形类型,包括普通行列报表、主/子报表、图形报表、交叉表等,以及条形图、饼图、曲线图、甘特图等多种图表类型,满足不同场景下的数据分析需求。 3. 报表发布管理:...

    Python Cookbook

    17. **3.17 在Unicode和普通字符串之间转换**:解释了如何处理Unicode字符串。 18. **3.18 将Unicode字符打印到标准输出**:提供了正确显示Unicode字符的方法。 19. **3.19 基于模式匹配的分发**:使用正则表达式...

    易语言数据库表格打印(易语言2003年大赛一等奖)-易语言

    易语言是一种简洁、高效、易于学习的编程语言,它的设计目标是降低计算机编程的门槛,让普通用户也能轻松掌握编程技能。2003年的易语言大赛中,一个名为“易语言数据库表格打印”的项目荣获了一等奖,这充分体现了其...

Global site tag (gtag.js) - Google Analytics