【问题标题】:Pivot table in MySQL - convert to pivot table with values in varchar formatMySQL 中的数据透视表 - 使用 varchar 格式的值转换为数据透视表
【发布时间】:2013-09-05 06:25:29
【问题描述】:

我想转换 Table1 中的数据,正如您在 Pivot_table 中数据的第一张图片上看到的那样。可以在 MySQL 中做到这一点吗?因为数据透视表(A,B,C,D)的值是varchar数据格式,我不能使用MySQL的任何聚合函数,如SUM或其他。

Table1:
PK        Name     Subject     Grade
-------------------------------------
1         Bob       Math        A
2         Bob       History     B
3         Bob       Language    C
4         Bob       Biology     D
5         Sue       History     C
6         Sue       Math        A
7         Sue       Music       A
8         Sue       Geography   C


Pivot_table:
Subject     Bob     Sue
-------------------------
Math        A        A
History     B        C
Language    C 
Biology     D
Music                A
Geography            C

感谢您的帮助

【问题讨论】:

  • 数据透视表是静态的吗?还是您即时创建?

标签: mysql sql pivot


【解决方案1】:

静态查询(就 Bob 和 Sue 而言)可能如下所示

SELECT subject, 
       MAX(CASE WHEN name = 'Bob' THEN grade END) `Bob`,
       MAX(CASE WHEN name = 'Sue' THEN grade END) `Sue` 
  FROM table1 
 GROUP BY subject

现在可以像这样使用动态 SQL 来解释其他名称

SET @sql = NULL;

SELECT GROUP_CONCAT(DISTINCT
         CONCAT('MAX(CASE WHEN name = ''', name,
                ''' THEN grade END) `', name, '`'))
  INTO @sql
  FROM table1;

SET @sql = CONCAT('SELECT subject, ', @sql, ' 
                     FROM table1 
                    GROUP BY subject');

PREPARE stmt FROM @sql;
EXECUTE stmt;
DEALLOCATE PREPARE stmt;

输出:

|主题 |鲍勃 |苏 | |-----------|--------|--------| |生物学 | D | (空) | |地理 | (空) | C | |历史 |乙| C | |语言 | C | (空) | |数学 |一个 |一个 | |音乐 | (空) |一个 |

这里是SQLFiddle演示


您可以将其包装到存储过程中以简化调用端的操作

DELIMITER $$
CREATE PROCEDURE sp_grade_report()
BEGIN
  SET @sql = NULL;

  SELECT GROUP_CONCAT(DISTINCT
           CONCAT('MAX(CASE WHEN name = ''', name,
                  ''' THEN grade END) `', name, '`'))
    INTO @sql
    FROM table1;

  SET @sql = CONCAT('SELECT subject, ', @sql, ' 
                       FROM table1 
                      GROUP BY subject');

  PREPARE stmt FROM @sql;
  EXECUTE stmt;
  DEALLOCATE PREPARE stmt;
END$$
DELIMITER ;

示例用法:

CALL sp_grade_report();

这里是SQLFiddle演示

【讨论】:

    【解决方案2】:

    为转换编写一个存储过程。

    1. SELECT DISTINCT name FROM table1; 获取 Pivot_table 的列列表。为主题创建具有适当列数 + 1 的表。

    2. 在数据透视表中插入所有主题以获得每个主题的 1 行。 INSERT ... from SELECT DISTINCT subject FROM table1.

    3. 创建游标以检索 table1 中的所有记录。遍历游标的所有行并为每一行调用UPDATE Pivot_table SET [COLUMN_NAME for studetn value]=[grade value] WHERE subject=[subject name value];

    【讨论】:

      猜你喜欢
      • 2015-01-26
      • 1970-01-01
      • 2012-12-06
      • 2018-04-10
      • 2015-12-22
      • 1970-01-01
      • 2020-08-28
      • 1970-01-01
      相关资源
      最近更新 更多