数据库:SQLServer 实现行转列、列转行用法笔记

数据库:SQLServer 实现行转列、列转行用法笔记



数据库:SQLServer 实现行转列、列转行用法笔记

在许多的互联网项目当中,报表开发是整个项目当中很重要的一个功能模块。其中会有一些比较复杂的报表统计需要行转列或者列转行的需求。今天给大家简单介绍一下在SQLServer当中如何使用PIVOT、UNPIVOT内置函数实现数据报表的行转列、列转行。有需要的朋友可以一起学习一下。


一、PIVOT、UNPIVOT用途

官方解释:可以使用 PIVOT 和 UNPIVOT 关系运算符将表值表达式更改为另一个表。PIVOT 通过将表达式某一列中的唯一值转换为输出中的多个列来旋转表值表达式,并在必要时对最终输出中所需的任何其余列值执行聚合。UNPIVOT 与 PIVOT 执行相反的操作,将表值表达式的列转换为列值。
注意:UNPIVOT运算符通过将列旋转到行来执行PIVOT的反向操作,UNPIVOT 并不完全是 PIVOT 的逆操作。PIVOT 执行聚合,并将多个可能的行合并为输出中的一行。UNPIVOT 不重现原始表值表达式的结果,因为行已被合并。另外,UNPIVOT 输入中的 NULL 值也在输出中消失了。如果值消失,表明在执行 PIVOT 操作前,输入中可能就已存在原始 NULL 值。

二、PIVOT语法格式

SELECT <非透视的列>,
    [第一个透视的列] AS <列名称>,
    [第二个透视的列] AS <列名称>,
    …
    [最后一个透视的列] AS <列名称>,
FROM
    (<生成数据的 SELECT 查询>)
    AS <源查询的别名>
PIVOT
(
    <聚合函数>(<要聚合的列>)
FOR
[<包含要成为列标题的值的列>]
    IN ( [第一个透视的列], [第二个透视的列],
    … [最后一个透视的列])
) AS <透视表的别名>
<可选的 ORDER BY 子句>;

三、行转列示例说明

-- 创建测试表 学习成绩统计表CREATE TABLE ScoreStatistics( UserName NVARCHAR(20), --学生姓名 SubjectName NVARCHAR(30), --科目名称 Score FLOAT, --成绩)-- 插入测试数据INSERT INTO ScoreStatistics SELECT '小王', '语文', 100INSERT INTO ScoreStatistics SELECT '小王', '数学', 90.5INSERT INTO ScoreStatistics SELECT '小王', '英语', 88INSERT INTO ScoreStatistics SELECT '小王', '历史', 65INSERT INTO ScoreStatistics SELECT '小李', '语文', 81INSERT INTO ScoreStatistics SELECT '小李', '数学', 99INSERT INTO ScoreStatistics SELECT '小李', '英语', 95INSERT INTO ScoreStatistics SELECT '小李', '历史', 90INSERT INTO ScoreStatistics SELECT '小刘', '语文', 90INSERT INTO ScoreStatistics SELECT '小刘', '数学', 85INSERT INTO ScoreStatistics SELECT '小刘', '英语', 59INSERT INTO ScoreStatistics SELECT '小刘', '历史', 98-- 传统写法select UserName, max(case SubjectName when '语文' then Score else 0 end)语文, max(case SubjectName when '数学'then Score else 0 end)数学, max(case SubjectName when '英语'then Score else 0 end)英语, max(case SubjectName when '历史'then Score else 0 end)历史from ScoreStatisticsgroup by UserName-- PIVOT 写法更简洁SELECT * FROM ScoreStatisticsAS PPIVOT( SUM(Score/*行转列后 列的值*/) FOR p.SubjectName/*需要行转列的列*/ IN ([语文],[数学],[英语],历史 /*列的值*/)) AS T-- order by 语文 desc 具体科目排序-- order by username desc -- 姓名排序-- 动态拼接列的示例DECLARE @sql_str VARCHAR(8000); -- 要执行的sql--拿到数值列 [历史],[数学],[英语],[语文]DECLARE @sql_col VARCHAR(8000);SELECT @sql_col = ISNULL(@sql_col + ',','') + QUOTENAME(SubjectName) FROM ScoreStatistics GROUP BY SubjectName;print(@sql_col); -- 打印数值列,不必需SET @sql_str = 'SELECT * FROM (SELECT [UserName],[SubjectName],[Score] FROM [ScoreStatistics]) p PIVOT(SUM([Score]) FOR [SubjectName] IN ( '+ @sql_col +') ) AS pvtORDER BY pvt.[UserName]'PRINT (@sql_str);--打印执行的sqlEXEC (@sql_str);-- 执行查询
输出结果
UserName 语文 数学 英语 历史
小王 100 90.5 88 65
小刘 90 85 59 98
小李 81 99 95 90


四、列转行示例

  
  
 
-- 插入测试表CREATE TABLE ScoreSummary( UserName NVARCHAR(20), --学生姓名 数学 FLOAT, --数学成绩 英语 FLOAT, --英语成绩 语文 FLOAT, --语文成绩 历史 FLOAT, --历史成绩)-- 插入测试数据INSERT INTO ScoreSummary SELECT '小李',81,99,95,90;INSERT INTO ScoreSummary SELECT '小刘',90,85,59,98;INSERT INTO ScoreSummary SELECT '小王',100,90.5,88,65;-- 查询用法select aa.UserName,aa.Scorefrom (select UserName,数学,英语,语文,历史 from dbo.ScoreSummary) as aunpivot(Score for ScoreSummary in(数学,英语,语文,历史)) as aa order by aa.UserName

输出结果

UserName Score
小李 81
小李 99
小李 95
小李 90
小刘 90
小刘 85
小刘 59
小刘 98
小王 100
小王 90.5
小王 88
小王 65

IT技术分享社区

个人博客网站:https://programmerblog.xyz

数据库:SQLServer 实现行转列、列转行用法笔记



文章推荐
SQL常用语句大全(值得收藏)
办公技巧:常用的100个Word快捷键!
GitHub上值得收藏的100个精选前端项目!
数据库优化:SQL 查找是否”存在”,别再 count 了,很耗费时间的!
学习 MySQL 高性能优化原理,这一篇就够了!
MySQL优化:数据量很大,分页查询很慢,有什么优化方案?
数据库:MySQL中,当update修改数据与原数据相同时会再次执行吗?



版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/2502.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Mysql日期和时间函数不求人

    Mysql日期和时间函数不求人

    2021年4月29日
    128
  • flash cookie的制作和使用例子详解 三

    flash cookie的制作和使用例子详解 三前面的两篇博客介绍的是怎么用页面来操作flashcookie,还要放在容器里运行,这篇做一个简单的仅仅使用flash就可以读写flashcookie的例子先看flash中的代码,当然这次要在flash中定义一些button显示,输入等控件,看页面就知道定义了哪些控件,再看代码就知道这些控件被命名成什么[img]http://dl2.iteye.com/upload/attac…

    2022年7月15日
    13
  • Java Serialized 序列化 作用

    今天在看设计模式时,看到迪米特法则时,有一点,谨慎使用Serialized:一个项目中使用远程方法调用方式传递一个VO(值对象),这个对象就必须实现Serialized接口,也就是把需要网络传输的对象进行序列化,否则会出现NotSerialiableException异常。若VO修改了一个属性的权限,由private改为public访问权限变大了,如果服务器上没有做出相应的变更,就会报…

    2022年4月4日
    65
  • c程序中整形变量只能存放整数实型变量只能存放浮点数_c语言合法的实型常量

    c程序中整形变量只能存放整数实型变量只能存放浮点数_c语言合法的实型常量vb中,以下变量类型1,数字型变量(numeric)2,字符串型变量(string)3,日期型变量(date)4,对象型变量(object)5,变体型变量(variant)这几个vb变量类型中,最最主要的就是前面两个,数字型变量和字符串型变量.意思很简单,数字型可以用来存放数字,字符串型存放文本.下面就来详细介绍这几种变量.1.数字型数字型变量有多种类型,在咱们的vb里,有3中数字数据类型1;整形…

    2022年10月26日
    0
  • Java不可重入锁和可重入锁理解[通俗易懂]

    Java不可重入锁和可重入锁理解[通俗易懂]最近正在阅读JavaReentrantLock源码,始终对可重入和不可重入概念理解不透彻,进行学习后记录在这里。基础知识Java多线程的wait()方法和notify()方法这两个方法是成对出现和使用的,要执行这两个方法,有一个前提就是,当前线程必须获其对象的monitor(俗称“锁”),否则会抛出IllegalMonitorStateException异常,所以这两个方法必须在同步…

    2022年6月26日
    22
  • leetcode 最长有效括号_字符指针赋值为什么不能加大括号

    leetcode 最长有效括号_字符指针赋值为什么不能加大括号给你一个只包含 ‘(’ 和 ‘)’ 的字符串,找出最长有效(格式正确且连续)括号子串的长度。示例 1:输入:s = “(()”输出:2解释:最长有效括号子串是 “()”示例 2:输入:s = “)()())”输出:4解释:最长有效括号子串是 “()()”示例 3:输入:s = “”输出:0题解括号匹配:(看作+1,)看作-1,所有满足条件的括号应该是前缀和>=0,并且总和==0class Solution {public: const int INF =

    2022年8月9日
    1

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号