MySql选择动态行值作为列名

时间:2014-08-20 17:28:09

标签: mysql sql pivot-table

这是我给出的表格:

+---------+------------+-------------------+--------------------------+---------------+---------------+
| exec_id | project_id | flow_id           | job_id                   | start_time    | end_time      |
+---------+------------+-------------------+--------------------------+---------------+---------------+
|   10919 |         16 | my_flow_cleanup   | init                     | 1408480308611 | 1408480308686 |
|   10919 |         16 | my_flow_cleanup   | job_id_1                 | 1408480309212 | 1408480309426 |
|   10919 |         16 | my_flow_cleanup   | job_id_2                 | 1408480308721 | 1408480308776 |
|   10919 |         16 | my_flow_cleanup   | job_id_3                 | 1408480308827 | 1408480309171 |
+---------+------------+-------------------+--------------------------+---------------+---------------+

我想要达到这样的选择:

exec_id     init_start            init_end            job_id_1_start       job_id_1_end       job_id_2_start        job_id_2_end        job_id_3_start      job_id_3_end
10919       1408480308611         1408480308686       1408480309212        1408480309426      1408480308721         1408480308776       1408480308827       1408480309171

我花了两天多的时间,但收效甚微。 This question帮助了我,但并非完全。如您所见,另一个问题涉及获取2列并将其转换为2行。但我必须采取3列并将它们转换为2行。

任何人都可以帮我吗?它甚至可以在MySQL中使用吗?

编辑1

感谢Khalid Junaid的回答,它解决了我的问题。我还要做一个修改。我必须通过start_time按升序对所选列进行排序。

例如:

+---------+------------+-------------------+----------+---------------+---------------+
| exec_id | project_id | flow_id           | job_id   | start_time    | end_time      |
+---------+------------+-------------------+----------+---------------+---------------+
|   10919 |         16 | my_flow_cleanup   | init     | 10            | 15            |
|   10919 |         16 | my_flow_cleanup   | job_id_1 | 30            | 40            |
|   10919 |         16 | my_flow_cleanup   | job_id_2 | 40            | 50            |
|   10919 |         16 | my_flow_cleanup   | job_id_3 | 20            | 25            |
+---------+------------+-------------------+----------+---------------+---------------+

当前查询将返回:

exec_id     init_start            init_end            job_id_1_start       job_id_1_end       job_id_2_start        job_id_2_end        job_id_3_start      job_id_3_end
10919       10                    15                  30                   40                 40                    50                  20                  25

我需要:

exec_id     init_start            init_end            job_id_3_start       job_id_3_end       job_id_1_start        job_id_1_end        job_id_2_start      job_id_2_end
10919       10                    15                  20                   25                 30                    40                  40                  50

请注意,根据start_time,列的顺序现在已更改。

我尝试使用temporary tablesviews执行此操作,我认为这很容易。不幸的是,我没有创建表/创建视图权限。有没有办法在没有临时表和视图的情况下实现这个目标?

2 个答案:

答案 0 :(得分:2)

从参考问题使用group_concat的方法开始,您可以这样做,但请注意一件事,因为每个exec_id组的工作ID会增加,因此group_concat方法不会是最优的要连接的默认长度为1024个字符,对于动态列,这肯定会越过该限制,但可以按照documentation

中的说明增加此限制
SET @sql = NULL;

SELECT GROUP_CONCAT(DISTINCT
  CONCAT('MAX(CASE WHEN job_id = ''',
         job_id, 
         ''' THEN start_time END) `',
         job_id,
         '_start`',
         ',',
         'MAX(CASE WHEN job_id = ''',
         job_id,
         ''' THEN end_time END) `',
         job_id,
         '_end`' 
         )

 )
  INTO @sql
  FROM t;

SET @sql = CONCAT('SELECT exec_id, ', @sql, ' 
                     FROM t 
                    GROUP BY exec_id');

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

Fiddle Demo

答案 1 :(得分:0)

是的,它可以在MySQL中使用,但是对于任何SQL数据库,您都无法进行动态从其找到的数据值创建列的查询。必须在准备查询时修复查询的列。因此,您必须事先知道要成为列的不同值。这可能需要另一个查询。在所有SQL数据库中都是如此,而不仅仅是MySQL。

SELECT exec_id,
 MIN(CASE job_id WHEN 'init' start_time END) AS init_start,
 MIN(CASE job_id WHEN 'init' end_time END) AS init_end,
 MIN(CASE job_id WHEN 'job_id_1' start_time END) AS job_id_1_start,
 MIN(CASE job_id WHEN 'job_id_1' end_time END) AS job_id_1_end
FROM `this_is_my_given_table`
GROUP BY exec_id

这基本上硬编码每个不同的工作ID,但这是必要的。