一,行转列

先建立测试数据

if OBJECT_ID('week_income') is not null
drop table week_income
go
create table week_income
(
employee varchar(10),
weekname varchar(10),
income int
)
go
insert into week_income
select '張三','星期一',1000 union all
select '張三','星期二',2000 union all
select '張三','星期三',1500 union all
select '張三','星期四',2000 union all
select '張三','星期五',3000 union all
select '張三','星期六',4000 union all
select '張三','星期日',5000 union all
select '李四','星期一',1000 union all
select '李四','星期二',2000 union all
select '李四','星期三',1500 union all
select '李四','星期四',2000 union all
select '李四','星期五',3000 union all
select '李四','星期六',4000 union all
select '李四','星期日',8000 select * from week_income

1.1 用传统的case when 语法实现行转列。

select employee as '社員'
,sum(case weekname when '星期一' then income else 0 end) as '星期一'
,sum(case weekname when '星期二' then income else 0 end) as '星期二'
,sum(case weekname when '星期三' then income else 0 end) as '星期三'
,sum(case weekname when '星期四' then income else 0 end) as '星期四'
,sum(case weekname when '星期五' then income else 0 end) as '星期五'
,sum(case weekname when '星期六' then income else 0 end) as '星期六'
,sum(case weekname when '星期日' then income else 0 end) as '星期日'
from week_income
group by employee

1.2 用sqlserver2005以后新增加的pivot语法实现行转列。

pivot的语法如下:

SELECT <non-pivoted column>,
[first pivoted column] AS <column name>,
[second pivoted column] AS <column name>,
...
[last pivoted column] AS <column name>
FROM
(<SELECT query that produces the data>)
AS <alias for the source query>
PIVOT
(
<aggregation function>(<column being aggregated>)
FOR
[<column that contains the values that will become column headers>]
IN ( [first pivoted column], [second pivoted column],
... [last pivoted column])
) AS <alias for the pivot table>
<optional ORDER BY clause>;
select employee  as '社員',[星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日] --step 3
from (select employee ,weekname,income from week_income) t -- step 2
pivot
(
    --step 1
sum(income) for weekname in([星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日])
) b

pivot分为3个步骤:

1,进行行列转换

sum(income) for weekname in([星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日])
注意:聚合函数sum等是决定重复行数据的处理方法,比如有两个星期一,就把这两个值合并了。 

2,定义检索数据源(select employee ,weekname,income from week_income) t

注意别名t不能省略。这里要特别注意employee这一列并没有在步骤1中出现,sqlserver会默认按employee进行分组,这个功能挺棒的。

3,选择结果集中的列,全部选择可以用*。

employee  as '社員',[星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日]
注意:可以对列名取别名,也可以只检索其中的部分列,而不必全部检索出所有列。

转换结果

二,列转行

以上面的转换结果作为数据表,我们先把结果导入到一张tmp的表中

select * into tmp from (
select employee as '社員',[星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日]
from (select employee ,weekname,income from week_income) t
pivot
(
sum(income) for weekname in([星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日])
) b
) c select * from tmp

然后,我们将星期一到星期日这几列转换到行上。

2.1 用传统的union all方法。

select 社員 as employee,'星期一' as weekname, 星期一 as income from tmp union all
select 社員 as employee,'星期二' as weekname, 星期二 as income from tmp union all
select 社員 as employee,'星期三' as weekname, 星期三 as income from tmp union all
select 社員 as employee,'星期四' as weekname, 星期四 as income from tmp union all
select 社員 as employee,'星期五' as weekname, 星期五 as income from tmp union all
select 社員 as employee,'星期六' as weekname, 星期六 as income from tmp union all
select 社員 as employee,'星期日' as weekname, 星期日 as income from tmp
order by 社員

2.2 用sqlserver2005以后新增的unpivot方法。

select 社員 as employee, weekname,income
from (select 社員,[星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日] from tmp) a
unpivot
(
income for weekname in ([星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日] )
) b

unpivot是pivot的逆操作,income和weekname都是需要创建的新列,income指代数据的值,weekname指代列名的值。

转换后的结果和我们最初建立的表一模一样

三,动态实现行列旋转

上面的方式,都是代码写了几个固定值旋转,没有实现动态列,扩展性不强。

要实现动态,主要使用拼sql的方式实现,以行转列的传统写法做个例子,其他的可以举一反三。

declare @sql as varchar(4000)
set @sql = 'select employee as ''社員''' select @sql = @sql + ',sum(case weekname when ''' + weekname + ''' then income else 0 end) as ''' + weekname + ''''
from (
select distinct weekname from week_income) a
set @sql =@sql + ' from week_income '
set @sql =@sql + ' group by employee '
print @sql
exec (@sql)

结果也是这样

sql行列旋转的更多相关文章

  1. SQL 行列倒置

    SQL的的行列倒置已经不是新知识了,但在博主的技术咨询期间,仍发现其实有很多人并不了解这块,所以在此专门写一篇博客记录.本文将以Mysql为例,并以数据采集指标信息获取为例子.在下面的例子,你可以在s ...

  2. SQL行列转换6种方法

    在进行报表开发时,很多时候会遇到行列转换操作,很对开发人员针对于SQL级别行列转换操作一直不甚理解,今天正好抽空对其进行了一些简单的总结.这里主要列举3种可以实现SQL行列转换的方法,包括通用SQL解 ...

  3. SQL行列乾坤大挪移

    “生活总是这样,有时候,你需要一个苹果,但别人却给了你一个梨.” 今天dalao邮件里需要添加一张每月累计长长的图,可是,拿到手上的SQL导出数据不符合我最爱的pyecharts的数据输入格式,头大. ...

  4. 数据透视表sql:用SQL行列转换实现数据透视的一些思考

    用SQL行列转换实现数据透视的一些思考 摘要:根据对报表开发过程中碰到的需要用SQL行列转换进行解决的一类查询统计问题的分析,逐步探索求解得到一种较通用的解决思路,并用函数进行实现.该解决思路及函数实 ...

  5. SQL行列轉換方法(詳細例子)

    普通行列转换(version 1.0)仅针对sql server 2000提供静态和动态写法,version 2.0增加sql server 2005的有关写法. 问题:假设有张学生成绩表(tb)如下 ...

  6. SQL表旋转

    在制作报表的时候,有时候会碰到基础资料是依照时间区间去一笔一笔记录的资料,但是使用者在看报表的时候想要将时间区间以横向的方式呈现不是直向的情况出现,又或者基础资料的表数据结构是横向的而使用者在看报表的 ...

  7. sql行列转换

    首先我们建立一张表,名为scoreInfo,各个字段的设计如下图,分别是name,course,score,表示姓名,成绩与分数,如图所示.

  8. mysql:sql行列转换

    今天一个同学遇到一个问题问我了,由于本人平时学习的mysql比较基础,确实没解决,后来google了一下,才知道是sql的一种技法[行列转换],话不多说先上图: 想得到下面的结果: +------+- ...

  9. SQL行列转换:报表_公司采购表_每个公司各采购了些什么产品

    有同学问了个比较典型行列转换的问题,想想,解答如下:数据库有一张表: 是个公司采购表,想转化成如下报表,显示每个公司各采购了些什么产品: 哪些公司采购哪些产品是不确定的,所以报表的列有哪几项是不确定的 ...

随机推荐

  1. this的问题

    javascript this可以绑定到:全局对象,自己定义的对象,用构造函数生成的对象,通过call或者apply更改绑定的对象    1.全局对象  1 2 3 4 5 function glob ...

  2. LINUX下查看日志

    LINUX的日志都在 /var/log 目录下:    进入此文件查看目录详情: 查看某个日志的命令: 1.cat messages可以查看某个日志文件. 2.要达到实时更新,可以通过tail命令查看 ...

  3. Thinkphp各大支付平台在线支付集成源码

    用Thinkphp给客户开发网站的时候需要用到各大平台付款功能,下面就免费分享给大家,此类是个成熟类,网上down下来的,经过修改测试了(可以直接拿来使用,附带使用方法,有需要的朋友请拿走.),如果有 ...

  4. 第二,C语言示例

    #include<stdio.h> int main (void)                 /*WTF*/ { int num; num=1; printf(" I am ...

  5. 24.Redis2.8主从集群sentinel

    0.集群架构(此处只说两种;本文2种,避免sentinel成为单节点) 第一种: 第二种: 1.下载redis2.8.x版本,2.8.x都是稳定版 redis-2.8.24.tar.gz 2.解压,安 ...

  6. .NET开源项目

      综合类 微软企业库 微软官方出品,是为了协助开发商解决企业级应用开发过程中所面临的一系列共性的问题, 如安全(Security).日志(Logging).数据访问(Data Access).配置管 ...

  7. application、viewstate、纯HTML提交方式

    Application - 全局公共变量组 存放位置:服务端 所有的访问用户都是访问的同一个变量 声明周期:永久 用法同session类似 viewstate-病例 因为http的无状态性,需要记录上 ...

  8. 基础笔记10(IO 1.7try-with-resource) 装饰模式

    1.读写的类型分为字节流和字符流,字节流一般是视频音频其他所有的类型都可以. (非文档文件使用字符流易造成未知编码(?)错误) InputStream OutputStream 抽象类 fileInp ...

  9. 注册码_EditPlus3

    1.来自"http://jingyan.baidu.com/article/67508eb4d78cfe9cca1ce488.html" Name:  www.cnzz.cc Co ...

  10. Java学习--内部类(一)

    Java学习--内部类(一) 一. 内部类的定义和特点 class Outer{ privite int num = 5; class Inner{ public void Display(){ Sy ...