web-dev-qa-db-fra.com

Mysql Convertir une colonne en ligne (tableau croisé dynamique)

J'ai une table comme ça

+---+-----+----+----+----+----+
|id |month|col1|col2|col3|col4|
+---+-----+----+----+----+----+
|101|Jan  |A   |B   |NULL|B   |
+---+-----+----+----+----+----+
|102|feb  |C   |A   |G   |E   |
+---+-----+----+----+----+----+

Et puis je veux créer un rapport comme celui-ci

+----+---+---+
|desc|jan|feb|
+----+---+---+
|col1|A  |C  |
+----+---+---+
|col2|B  |A  |
+----+---+---+
|col3|0  |G  |
+----+---+---+
|Col4|B  |E  |
+----+---+---+

Quelqu'un peut-il m'aider?

19
user1914516

Ce que vous devez faire, c'est d'abord, débloquer les données, puis les faire pivoter. Mais malheureusement, MySQL n'a pas ces fonctions, vous devrez donc les répliquer à l'aide d'un UNION ALL requête pour le pivot et une fonction d'agrégation avec un CASE pour le pivot.

L'impivot ou UNION ALL piece prend les données de votre col1, col2, etc. et les transforme en plusieurs lignes:

select id, month, col1 value, 'col1' descrip
from yourtable
union all
select id, month, col2 value, 'col2' descrip
from yourtable
union all
select id, month, col3 value, 'col3' descrip
from yourtable
union all
select id, month, col4 value, 'col4' descrip
from yourtable

Voir SQL Fiddle with Demo .

Résultat:

|  ID | MONTH |  VALUE | DESCRIP |
----------------------------------
| 101 |   Jan |      A |    col1 |
| 102 |   feb |      C |    col1 |
| 101 |   Jan |      B |    col2 |
| 102 |   feb |      A |    col2 |
| 101 |   Jan | (null) |    col3 |
| 102 |   feb |      G |    col3 |
| 101 |   Jan |      B |    col4 |
| 102 |   feb |      E |    col4 |

Vous encapsulez ensuite cela dans une sous-requête pour appliquer l'agrégat et le CASE pour le convertir au format souhaité:

select descrip, 
  max(case when month = 'jan' then value else 0 end) jan,
  max(case when month = 'feb' then value else 0 end) feb
from
(
  select id, month, col1 value, 'col1' descrip
  from yourtable
  union all
  select id, month, col2 value, 'col2' descrip
  from yourtable
  union all
  select id, month, col3 value, 'col3' descrip
  from yourtable
  union all
  select id, month, col4 value, 'col4' descrip
  from yourtable
) src
group by descrip

Voir SQL Fiddle avec démo

Le résultat est:

| DESCRIP | JAN | FEB |
-----------------------
|    col1 |   A |   C |
|    col2 |   B |   A |
|    col3 |   0 |   G |
|    col4 |   B |   E |
40
Taryn