zoukankan      html  css  js  c++  java
  • 使用 PIVOT 和 UNPIVOT实现查询结果行转列

    语法:

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

    模拟数据:

     INSERT INTO WEEK_INCOME
    SELECT '星期一',1000
    UNION ALL
    SELECT '星期二',2000
    UNION ALL
    SELECT '星期三',3000
    UNION ALL
    SELECT '星期四',4000
    UNION ALL
    SELECT '星期五',5000
    UNION ALL
    SELECT '星期六',6000
    UNION ALL
    SELECT '星期日',7000

    一般我们最经常使用的查询是查询一周中每天或某几天的收入,例如查询周一至周日全部的收入:

    SELECT WEEK,INCOME FROM WEEK_INCOME

    得到如下的查询结果集:

    WEEK           INCOME
    星期一           1000
    星期二           2000
    星期三           3000
    星期四           4000
    星期五           5000
    星期六           6000
    星期日           7000

    但是在一些情况下(往往是某些报表中),我们希望在一行中显示周一至周日的收入,这时候查询结果集应该是这样的:

    星期一   星期二   星期三   星期四   星期五   星期六   星期日
    1000     2000     3000     4000     5000     6000     7000

    这种情况下,SQL查询语句可以这样写:

      SELECT
    SUM(CASE WEEK WHEN '星期一' THEN INCOME END) AS [星期一],
    SUM(CASE WEEK WHEN '星期二' THEN INCOME END) AS [星期二],
    SUM(CASE WEEK WHEN '星期三' THEN INCOME END) AS [星期三],
    SUM(CASE WEEK WHEN '星期四' THEN INCOME END) AS [星期四],
    SUM(CASE WEEK WHEN '星期五' THEN INCOME END) AS [星期五],
    SUM(CASE WEEK WHEN '星期六' THEN INCOME END) AS [星期六],
    SUM(CASE WEEK WHEN '星期日' THEN INCOME END) AS [星期日]
    FROM WEEK_INCOME

    但是,在SQL SERVER 2005中提供了更为简便的方法,这就是"PIVOT"关系运算符。(相反的“列转行”是UNPIVOT),以下是使用PIVOT实现“行转列”的SQL语句

    SELECT [星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日]
    FROM WEEK_INCOME
    PIVOT
    (
        SUM(INCOME) for [week] in([星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日])
    )TBL

    正常情况下的查询结果是这样:

    星期一           1000
    星期二           2000
    星期三           3000
    星期四           4000
    星期五           5000
    星期六           6000
    星期日           7000

    行转列后是这样:

    星期一   星期二   星期三   星期四   星期五   星期六   星期日
    1000    2000    3000    4000    5000    6000    7000

    也就是说,行转列后,原来的某个列的值变做了列名,在这里就是原来WEEK列的值“星期一”,"星期二"..."星期日"边做了列名,而我们需要做的另一个工作就是计算这些列的值(这里的“计算”其实就是PIVOT里面的聚合函数(sum,avg等))

    SELECT [星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日]--这里是PIVOT第三步(选择行转列后的结果集的列)这里可以用“*”表示选择所有列,也可以只选择某些列(也就是某些天)
    FROM WEEK_INCOME --这里是PIVOT第二步骤(准备原始的查询结果,因为PIVOT是对一个原始的查询结果集进行转换操作,所以先查询一个结果集出来)这里可以是一个select子查询,但为子查询时候要指定别名,否则语法错误
    PIVOT
    (
        SUM(INCOME) for [week] in([星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日])--这里是PIVOT第一步骤,也是核心的地方,进行行转列操作。聚合函数SUM表示你需要怎样处理转换后的列的值,是总和(sum),还是平均(avg)还是min,max等等。例如如果week_income表中有两条数据并且其week都是“星期一”,其中一条的income是1000,另一条income是500,那么在这里使用sum,行转列后“星期一”这个列的值当然是1500了。后面的for [week] in([星期一],[星期二]...)中 for [week]就是说将week列的值分别转换成一个个列,也就是“以值变列”。但是需要转换成列的值有可能有很多,我们只想取其中几个值转换成列,那么怎样取呢?就是在in里面了,比如我此刻只想看工作日的收入,在in里面就只写“星期一”至“星期五”(注意,in里面是原来week列的值,"以值变列")。总的来说,SUM(INCOME) for [week] in([星期一],[星期二],[星期三],[星期四],[星期五],[星期六],[星期日])这句的意思如果直译出来,就是说:将列[week]值为"星期一","星期二","星期三","星期四","星期五","星期六","星期日"分别转换成列,这些列的值取income的总和。
    )TBL--别名一定要写
  • 相关阅读:
    java Activiti 工作流引擎 SSM 框架模块设计方案
    自定义表单 Flowable 工作流 Springboot vue.js 前后分离 跨域 有代码生成器
    数据库设计的十个最佳实践
    activiti 汉化 stencilset.json 文件内容
    JAVA oa 办公系统模块 设计方案
    java 考试系统 在线学习 视频直播 人脸识别 springboot框架 前后分离 PC和手机端
    集成 nacos注册中心配置使用
    “感恩节 ”怼记
    仓颉编程语言的一点期望
    关于System.out.println()与System.out.print("\n")的区别
  • 原文地址:https://www.cnblogs.com/birkhoff/p/6170840.html
Copyright © 2011-2022 走看看