一、Pivot和UnPivot介绍
1.Pivot介绍
PIVOT用于将列值旋转为列名(即行转列),在SQL Server 2000可以用聚合函数配合CASE语句实现

PIVOT的一般语法是:PIVOT(聚合函数(列) FOR 列 in (…) )AS P

完整语法:

table_source

PIVOT(

聚合函数(value_column)

FOR pivot_column

IN(<column_list>)

)
2.UnPivot介绍
UNPIVOT用于将列明转为列值(即列转行),在SQL Server 2000可以用UNION来实现

完整语法:

table_source

UNPIVOT(

value_column

FOR pivot_column

IN(<column_list>)

)

注意:PIVOT、UNPIVOT是SQL Server 2005 的语法,使用需修改数据库兼容级别  在数据库属性->选项->兼容级别改为   90

二、行转列实现

1、建立表格

if object_id('TestTB') is not null
drop table TestTB
go create table TestTB(姓名 varchar(10),课程 varchar(10),分数 int) insert into TestTB values('张三','语文',74)
insert into TestTB values('张三','数学',83)
insert into TestTB values('张三','物理',93)
insert into TestTB values('李四','语文',74)
insert into TestTB values('李四','数学',84)
insert into TestTB values('李四','物理',94)
go select * from TestTB

姓名       课程       分数

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

张三       语文        74

张三       数学        83

张三       物理        93

李四       语文        74

李四       数学        84

李四       物理        94

2、使用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) 物理
from TestTB
group by 姓名

姓名       语文        数学        物理

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

李四        74          84          94

张三        74          83          93

3、使用SQL Server 2000动态SQL

--SQL SERVER 2000动态SQL,指课程不止语文、数学、物理这三门课程。(以下同)
--变量按sql语言顺序赋值
declare @sql varchar(500)
set @sql='select 姓名'
select @sql=@sql+',max(case 课程 when '''+课程+''' then 分数 else 0 end)['+课程+']'
from(select distinct 课程 from TestTB)a--同from TestTB group by课程,默认按课程名排序
set @sql=@sql+' from TestTB group by 姓名'
exec(@sql) --使用isnull(),变量先确定动态部分
declare @sql varchar(8000)
select @sql=isnull(@sql+',','')+' max(case 课程 when '''+课程+''' then 分数 else 0 end) ['+课程+']'
from(select distinct 课程 from TestTB) as a
set @sql='select 姓名,'+@sql+' from TestTB group by 姓名'
exec(@sql)

姓名       数学        物理        语文

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

李四        84          94          74

张三        83          93          74

4、使用SQL Server 2005静态SQL

select * from TestTB pivot(max(分数) for 课程 in(语文,数学,物理))a

5、使用SQL Server 2005动态SQL

--使用stuff()
declare @sql varchar(8000)
set @sql='' --初始化变量@sql
select @sql=@sql+','+课程 from TestTB group by 课程--变量多值赋值
set @sql=stuff(@sql,1,1,'')--去掉首个','
set @sql='select * from TestTB pivot (max(分数) for 课程 in ('+@sql+')) a'
exec(@sql) --或使用isnull()
declare @sql varchar(8000)
--获得课程集合
select @sql=isnull(@sql+',','')+课程 from TestTB group by 课程
set @sql='select * from TestTB pivot (max(分数) for 课程 in ('+@sql+')) a'
exec(@sql)

三、行转列结果加上总分、平均分

1、使用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) 物理,
sum(分数)总分,
cast(avg(分数*1.0) as decimal(18,2))平均分
from TestTB
group by 姓名

姓名       语文        数学        物理        总分        平均分

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

李四        74          84          94          252         84.00

张三        74          83          93          250         83.33

2、使用SQL Server 2000 动态SQL

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

3、使用SQL Server 2005静态SQL

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

4、使用SQL Server 2005动态SQL

--使用stuff()
declare @sql varchar(8000)
set @sql='' --初始化变量@sql
select @sql=@sql+','+课程 from TestTB group by 课程--变量多值赋值
--同select @sql = @sql + ','+课程 from (select distinct 课程 from TestTB)a
set @sql=stuff(@sql,1,1,'')--去掉首个','
set @sql='select m.* , n.总分,n.平均分 from
(select * from (select * from TestTB) a pivot (max(分数) for 课程 in ('+@sql+')) b) m ,
(select 姓名,sum(分数) 总分, cast(avg(分数*1.0) as decimal(18,2)) 平均分 from TestTB group by 姓名) n
where m.姓名= n.姓名'
exec(@sql) --或使用isnull()
declare @sql varchar(8000)
select @sql=isnull(@sql+',','')+课程 from TestTB group by 课程
set @sql='select m.* , n.总分,n.平均分 from
(select * from (select * from TestTB) a pivot (max(分数) for 课程 in ('+@sql+')) b) m ,
(select 姓名,sum(分数)总分, cast(avg(分数*1.0) as decimal(18,2)) 平均分 from TestTB group by 姓名) n
where m.姓名= n.姓名'
exec(@sql)

四、列转行实现

1、建立表格

if object_id('TestTB') is not null
drop table TestTB
go
create table TestTB( 姓名 varchar(10),语文 int,数学 int,物理 int)
insert into TestTB values('张三',74,83,93)
insert into TestTB values('李四',74,84,94)
go
select * from TestTB
go

姓名       语文        数学        物理

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

张三       74          83          93

李四        74          84          94

2、使用SQL Server 2000静态SQL

select * from
(
select 姓名,课程='语文',分数=语文 from TestTB
union all
select 姓名,课程='数学',分数=数学 from TestTB
union all
select 姓名,课程='物理',分数=物理 from TestTB
) as temp
order by 姓名,case 课程 when '语文' then 1 when '数学' then 2 when '物理' then 3 end

姓名       课程 分数

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

李四       语文 74

李四       数学 84

李四       物理 94

张三       语文 74

张三       数学 83

张三       物理 93

3、使用SQL Server 2000动态SQL

declare @sql varchar(8000)
select @sql=isnull(@sql+' union all ','')+' select 姓名, [课程]='
+quotename(Name,'''')+' , [分数] = '+quotename(Name)+' from TestTB' from syscolumns
where Name!='姓名' and ID=object_id('TestTB')--表名TestTB,不包含列名为姓名的其他列
order by colid
exec(@sql+' order by 姓名')
go

4、使用SQL Server 2005静态SQL

select 姓名,课程,分数 from TestTB unpivot (分数 for 课程 in([语文],[数学],[物理])) temp

5、使用SQL Server 2005动态SQL

declare @sql nvarchar(4000)
select @sql=isnull(@sql+',','')+quotename(Name) from syscolumns
where ID=object_id('TestTB') and Name not in('姓名') order by Colid
set @sql='select 姓名,[课程],[分数] from TestTB unpivot ([分数] for [课程] in('+@sql+'))b'
exec(@sql)

博客转自:luluping的《SQLServer行列转换 Pivot UnPivot

[转载]SQL Server行列转换实现的更多相关文章

  1. SQL Server 行列转换

    /* 标题:普通行列转换(version 2.0) 作者:范中磊 说明:普通行列转换(version 1.0)仅针对sql server 2000提供静态和动态写法,version 2.0增加sql ...

  2. 12、SQL Server 行列转换

    SQL Server 行转列 在SQL Server 2005中PIVOT 用于将列值转换为列名(行转列),在SQL Server 2000中是没有这个关键字的 只能用case语句实现. --创建测试 ...

  3. SQL SERVER 行列转换(动态)

    行转列测试数据: --测试数据 if not object_id(N'Tempdb..#T') is null drop table #T Go Create table #T([Name] nvar ...

  4. 转载——SQL Server数据库性能优化之SQL语句篇

    转载自:http://www.blogjava.net/allen-zhe/archive/2010/07/23/326927.html 1. 按需索取字段,跟“SELECT *”说拜拜 字段的提取一 ...

  5. Sql的行列转换

    创建表scores 一.传统的行列转换 纵表转横表 我们要转成的横表是这样子的: pivot是sql server 2005 提供的运算符,所以只要数据库在05版本以上的都可以使用.主要用于行和列的转 ...

  6. Sql实现行列转换

    从MS Sql Server 2005微软就推出了pivot和unpivot实现行列转换,这极大的方便了我们存储数据和呈现数据.今天就对这两个关键字进行分析,结合实例讲解如何存储数据,如何呈现数据. ...

  7. SQL Server数据库转换成oracle

    来源:http://blog.csdn.net/hzfu007/article/details/6182151 经常碰到需要把sql server的数据迁移到Oracle的情况. 在网上查找一下,有很 ...

  8. 转载 SQL Server中索引管理之六大铁律

    转载原地址 http://jingyan.baidu.com/article/48a42057c03bd7a924250429.html 索引是以表列为基础的数据库对象.索引中保存着表中排序的索引列, ...

  9. [转载]sql server 等待类型

    下表列出各任务所遇到的等待类型. 等待类型 说明 ASYNC_DISKPOOL_LOCK 当尝试同步并行的线程(执行创建或初始化文件等任务)时出现. ASYNC_IO_COMPLETION 当某任务正 ...

随机推荐

  1. 学习笔记之SQL 教程

    SQL 教程 | 菜鸟教程 http://www.runoob.com/sql/sql-tutorial.html SQL,指结构化查询语言,全称是 Structured Query Language ...

  2. Java之POI的excel导入导出

    一.Apache POI是一种流行的API,它允许程序员使用Java程序创建,修改和显示MS Office文件.这由Apache软件基金会开发使用Java分布式设计或修改Microsoft Offic ...

  3. 十一.jQuery源码解析之.pushStack()

    pushStack()顾明思意,就是像桟中添加东西呗,现在看看他是如何添加东西的. 创建一个空的jQuery对象,然后把Dom元素集合放入这个jQuery对象中, 并保留对当前jQuery对象的引用. ...

  4. Python中小整数对象池和大整数对象池

    1. 小整数对象池 整数在程序中的使用非常广泛,Python为了优化速度,使用了小整数对象池, 避免为整数频繁申请和销毁内存空间. Python 对小整数的定义是 [-5, 256] 这些整数对象是提 ...

  5. 精《Linux内核精髓:精通Linux内核必会的75个绝技》一HACK #8 调度策略

    HACK #8 调度策略 本节介绍Linux的调度策略(scheduling policy).Linux调度策略的类别大致可以分为TSS(Time Sharing System,分时系统)和实时系统这 ...

  6. SpringBoot 之 普通类获取Spring容器中的bean

    [十]SpringBoot 之 普通类获取Spring容器中的bean   我们知道如果我们要在一个类使用spring提供的bean对象,我们需要把这个类注入到spring容器中,交给spring容器 ...

  7. django-给外键关系传值,删除外键关系

    反查: 在表关系里 related_name = '反查name',自己不设置,django也会默认设置为class的小写名字+_set  , ex: book_set. 一对一关系赋值: class ...

  8. python beautifulsoup爬虫

    爬虫这个听起来很 hack 的名字,是我学习 python 的诱因.当 python 基础学习到一定程度(基本语法,数据类型掌握) 就可以开启自己的小爬虫了.毕竟实践才是提高的最快途径.废话说完了,下 ...

  9. c#,条码

    static void Generate2(string text){ BarcodeWriter writer = new BarcodeWriter(); //使用ITF 格式,不能被现在常用的支 ...

  10. Flume学习总结

    Flume学习总结 flume是一个用来采集数据的软件,它可以从数据源采集数据到一个集中存放的地方. 最常用flume的数据采集场景是对日志的采集,不过,lume也可以用来采集其他的各种各样的数据,因 ...