MSSQL纵列转横列
在工作中我们一般会遇到将纵列转横列的需求,具体代码:
1.建表
CREATE TABLE [dbo].[AcrossChangeEndLong](
[Id] [int] IDENTITY(1,1) NOT NULL,
[Name] [nvarchar](50) NOT NULL,
[Subject] [nvarchar](50) NOT NULL,
[Score] [int] NOT NULL,
CONSTRAINT [PK_AcrossChangeEndLong] PRIMARY KEY CLUSTERED
(
[Id] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY] GO
SET IDENTITY_INSERT [dbo].[AcrossChangeEndLong] ON GO
INSERT [dbo].[AcrossChangeEndLong] ([Id], [Name], [Subject], [Score]) VALUES (1, N'张三', N'语文', 100)
GO
INSERT [dbo].[AcrossChangeEndLong] ([Id], [Name], [Subject], [Score]) VALUES (3, N'李四', N'语文', 80)
GO
INSERT [dbo].[AcrossChangeEndLong] ([Id], [Name], [Subject], [Score]) VALUES (4, N'张三', N'英语', 80)
GO
INSERT [dbo].[AcrossChangeEndLong] ([Id], [Name], [Subject], [Score]) VALUES (5, N'李四', N'英语', 50)
GO
INSERT [dbo].[AcrossChangeEndLong] ([Id], [Name], [Subject], [Score]) VALUES (6, N'张三', N'数学', 55)
GO
INSERT [dbo].[AcrossChangeEndLong] ([Id], [Name], [Subject], [Score]) VALUES (7, N'李四', N'数学', 34)
GO
SET IDENTITY_INSERT [dbo].[AcrossChangeEndLong] OFF
GO
ALTER TABLE [dbo].[AcrossChangeEndLong] ADD CONSTRAINT [DF_AcrossChangeEndLong_Score] DEFAULT ((0)) FOR [Score]
GO
2.具体SQL
SELECT Name AS '姓名',
MAX(CASE [Subject]
WHEN '语文' THEN Score
ELSE 0
END) AS '语文' ,
MAX(CASE [Subject]
WHEN '英语' THEN Score
ELSE 0
END) AS '英语' ,
MAX(CASE [Subject]
WHEN '数学' THEN Score
ELSE 0
END) AS '数学'
FROM dbo.AcrossChangeEndLong
GROUP BY Name ORDER BY Name
如果到时候增加了科目,比如增加了化学,这时候为了再次修改,我们可以弄成动态的(根据列自动增加),这里使用的是动态拼接SQL,会根据科目的增加而增加列,具体SQL如下
DECLARE @sql VARCHAR(8000)
SET @sql = 'SELECT [Name],'
SELECT @sql = @sql + 'SUM(CASE [Subject] WHEN ''' + [Subject]
+ ''' THEN [Score] ELSE 0 END) AS ''' + [Subject] + ''','
FROM ( SELECT DISTINCT
[Subject]
FROM dbo.AcrossChangeEndLong
) AS a
SELECT @sql = LEFT(@sql, LEN(@sql) - 1)
+ ' FROM [AcrossChangeEndLong] GROUP BY [Name]'
PRINT ( @sql )
EXEC(@sql)
另外在SQL Server 2005之后有了一个专门的PIVOT 和 UNPIVOT 关系运算符做行列之间的转换
注意:这种不能用于 eg: CASE [Subject] WHEN '数学' THEN Score ELSE 0 END 如果Then 后面是1(统计的时候会用到)而不是具体字段的时候,使用会报错,具体SQL如下
SELECT *
FROM ( SELECT Name AS '姓名',
Subject ,
Score
FROM dbo.AcrossChangeEndLong
) p PIVOT ( MAX(Score) FOR Subject IN ( [数学], [英语], [语文] ) ) AS pvt
ORDER BY pvt.姓名
使用Pivot动态拼接SQL:
DECLARE @sql_str VARCHAR(MAX)
DECLARE @sql_col VARCHAR(MAX)
SELECT @sql_col = ISNULL(@sql_col + ',', '') + QUOTENAME([Subject])
FROM dbo.AcrossChangeEndLong
GROUP BY [Subject]
SET @sql_str = '
SELECT * FROM (
SELECT [Name],[Subject],[Score] FROM [AcrossChangeEndLong]) p PIVOT
(SUM([Score]) FOR [Subject] IN ( ' + @sql_col + ') ) AS pvt
ORDER BY pvt.[Name]'
PRINT ( @sql_str )
EXEC (@sql_str)
具体效果:

扩展:可能我们需要加一个统计行,可以用UNION ALL连接 上一个表的结果【可以将结果插入临时表】,再用函数SUM求和各个列
MSSQL纵列转横列的更多相关文章
- (转载)SQL语句,纵列转横列
SQL语句,纵列转横列 Feed: 大富翁笔记 Title: SQL语句,纵列转横列 Author: wzmbox Comments sTable.db库位 货物编号 库存数1 0101 501 01 ...
- MSSQL横列转纵列
上篇我们说到了纵列转横列,这篇讲下横列转纵列,具体代码: 1.建表 CREATE TABLE [dbo].[EndLongChangeAcross]( ,) NOT NULL, ) NOT NULL, ...
- CSS实现横列布局的方法总结
一.使用float实现横列布局的方法 如下面所示:DIV1和DIV2都可以选择向左或者向右浮动50%来实现展示在同一行 div1 div2 实现下面图片中布局的css样式如下: 分析: 1.第一行第一 ...
- mssql Sqlver 修改标识列方法
摘自: http://www.cnblogs.com/badboy2008/articles/1145465.html MSSQL Server修改标识列方法 ----允许对系统表进行更新exec ...
- 一、CSS实现横列布局的方法总结
一.使用float实现横列布局的方法 如下面所示:DIV1和DIV2都可以选择向左或者向右浮动50%来实现展示在同一行 div1 div2 实现下面图片中布局的css样式如下: 分析: 1.第一行第一 ...
- mssql 获取自增列起始及增量
--首先创建一个表 CREATE TABLE [dbo].[abcd]( ,) NOT NULL, ) NULL, ) NULL ) ON [PRIMARY] --获取起始值 SELECT IDENT ...
- MSSQL—按照某一列分组后取前N条记录
以前在开发的时候遇到过一个需求,就是要按照某一列进行分组后取前几条数据,今天又有同事碰到了,帮解决了之后顺便写一篇博客记录一下. 首先先建一个基础数据表,代码如下: IF OBJECT_ID(N'Te ...
- mssql 动态行转列。
)) ,'张三' ,'李四' ,'王五' select * from #a a b ----------- ---- 张三 李四 王五 ( 行受影响) --行转列,步骤:''+张三+],[+王五+], ...
- mssql sqlserver 表增加列后,视图不会自动更新相关列的两种解决方法分享
摘要: 今天对物理数据表,进行增加列操作后,程序一直显示无法找到相应列,通过仔细比对发现,视图中无相应列更新,下文将具体的解决方法分享如下: 例: create view vw_test as sel ...
随机推荐
- 树&二叉树&哈夫曼树
1.树 需要注意的两点:n(n>=0)表示结点的个数,m表示子树的个数 (1)n>0时,树的根节点是唯一的. (2)m>0时,子树的个数没有限制. 结点的度和树的度 (1)结点的度是 ...
- PJSIP-PJLIB(samples) (the usage of the pjlib lib) (eg:string/I/O)
Here are some samples about PJLIB! PJLIB is the basic lib of PJSIP, so we need master the lib first ...
- linux统计分析流量-wireshark
wireshark是一款带界面的开源抓包工具,可以用来对系统流量进行统计分析. 安装 由于wireshark是带界面的,所以一般在界面环境下运行,可以通过yum安装: $ yum install -y ...
- 【Binary Search Tree Iterator 】cpp
题目: Implement an iterator over a binary search tree (BST). Your iterator will be initialized with th ...
- ssm项目中ueditor富文本编辑器的使用
一.下载 https://ueditor.baidu.com/website/index.html 将ueditor放到项目中合适的位置 二 . 配置文件上传路径 在utf8-jsp/jsp/conf ...
- How to solve SyntaxError on autogenerated manage.py?
错误原因: 今天尝试使用pycharm中的Django. 使用的是基于python3.5的Django2.0.5. 按照官网教程https://docs.djangoproject.com/en/2. ...
- Linux开启MySQL远程连接
Linux开启MySQL远程连接的设置步骤 . MySQL默认root用户只能本地访问,不能远程连接管理MySQL数据库,那么Linux下如何开启MySQL远程连接?设置步骤如下: 1.GRANT命令 ...
- HDU 4714 Tree2cycle 找规律
假设最少删除的边的个数为cost,显然,最终答案即为cost+cost+1 (因为删除一条边,就会增加一个链,所以删除cost条边后,就会有cost+1条链,将这cost+1条链连接起来的代价为cos ...
- PHP面向对象练习
练习内容:随机生成一个字符串 代码: <?phpclass randstring{ private $length; private $type; private $one = array(0, ...
- 【bzoj5017】[Snoi2017]炸弹 线段树优化建图+Tarjan+拓扑排序
题目描述 在一条直线上有 N 个炸弹,每个炸弹的坐标是 Xi,爆炸半径是 Ri,当一个炸弹爆炸时,如果另一个炸弹所在位置 Xj 满足: Xi−Ri≤Xj≤Xi+Ri,那么,该炸弹也会被引爆. 现在 ...