转载:http://www.cnblogs.com/newwind521/archive/2010/11/25/1887203.html

/* 
标题:普通行列转换(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 = 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 行专列 列转行 普通行列转换的更多相关文章

  1. SQL行转列,列转行

    SQL 行转列,列转行 行列转换在做报表分析时还是经常会遇到的,今天就说一下如何实现行列转换吧. 行列转换就是如下图所示两种展示形式的互相转换 行转列 假如我们有下表: SELECT * FROM s ...

  2. SQL 行转列和列转行

    SQL 行转列和列转行 行列互转,是一个经常遇到的需求.实现的方法,有case when方式和2005之后的内置pivot和unpivot方法来实现. 在读了技术内幕那一节后,虽说这些解决方案早就用过 ...

  3. sql 行转列 PIVOT 列转行 UNPIVOT

    原文:sql 行转列 PIVOT 列转行 UNPIVOT 一: 现有表一(t_table1),想转为表二(t_table2)的格式. 表一: 年 公司 收入 2013 公司1 12 2013 公司2 ...

  4. Ms sql行转列。汇总

    SQL行转列汇总 PIVOT 用于将列值旋转为列名(即行转列),在 SQL Server 2000可以用聚合函数配合CASE语句实现 PIVOT 的一般语法是:PIVOT(聚合函数(列) FOR 列 ...

  5. SQL行转列汇总

    PIVOT用于将列值旋转为列名(即行转列),在SQL Server 2000可以用聚合函数配合CASE语句实现 PIVOT的一般语法是:PIVOT(聚合函数(列) FOR 列 in (…) )AS P ...

  6. sql 行转列总结

    原文:sql 行转列总结 PIVOT UNPIVOT的用法 PIVOT用于将列值旋转为列名(即行转列),在SQL Server 2000可以用聚合函数配合CASE语句实现 PIVOT的一般语法是:PI ...

  7. 转 SQL行转列汇总

    1.PIVOT 用于将列值旋转为列名(即行转列) PIVOT 的一般语法是:PIVOT(聚合函数(列名) FOR 列名 in (列值1,…) )AS P select * from TB pivot ...

  8. SQL行转列 (及EAV模型获取数据)

    参考文章: http://www.williamsang.com/archives/1508.html 情景简介 学校里面记录成绩,每个人的选课不一样,而且以后会添加课程,所以不需要把所有课程当作列. ...

  9. sql 行转 列, 列转行

    行列互转 复制代码 create table test(id ),quarter int,profile int) insert into test values(,,) insert into te ...

随机推荐

  1. html页面的绝对路径和相对路径

    在用springmvc架构开发网站的过程中,离不开开发前台html页面,html经常需要使用本地相关的资源,如:图片,js,css等,一般情况下,我们可以通过使用相对路径的方式来对这些资源进行指向和访 ...

  2. iOS——为Xcode编译POCO C++静态库

    一.POCO C++ library简介 POCO C++ library是一个C++编写的跨平台库,主要实现网络连接.数据库管理以及服务器,适用于跨平台.嵌入式. 二.为Xcode编译POCO C+ ...

  3. ORA-12705: Cannot access NLS data files or invalid environment specified

    ASM实例无法启动 [grid@data ~]$ sqlplus / as sysasm SQL*Plus: Release 11.2.0.4.0 Production on Fri Sep 11 0 ...

  4. RequireJS加载ArcGIS API for JavaScript

    1.在main.js中配置ArcGIS API for JavaScript require.config({ paths : { //arcgisJS "esri": " ...

  5. git tag之后如何修改

    先 git clone 整个仓库,然后 git checkout tag_name 就可以取得 tag 对应的代码了. 但是这时候 git 可能会提示你当前处于一个“detached HEAD&quo ...

  6. php imagecreatetruecolor输出字符符或验证码

    $img = imagecreatetruecolor(100,100); //创建真彩图像资源 $color = imagecolorAllocate($img,200,200,200); //分配 ...

  7. 一个最简html5文档来说明html5的新特性和写法

    <!DOCTYPE html> <html lang="zh"> <head> <meta charset="utf-8&quo ...

  8. C#抓包

    抓包過程中發現頁面<input type="hidden" name="__LASTFOCUS" id="__LASTFOCUS" v ...

  9. discuz核心类库class_core的函数注释

    class discuz_core { // 数据库存储引擎 var $db = null; // 内存缓冲object var $mem = null; // 会话 object var $sess ...

  10. hdu 3853LOOPS (概率DP)

    LOOPS Time Limit: 15000/5000 MS (Java/Others)    Memory Limit: 125536/65536 K (Java/Others) Total Su ...