一:列转行
现在有一个用户属性表,它包含以下字段:
主键ID(id),用户属性(key),用户属性值(val),用户ID(owner)

原始数据如下图所示:

MySQL转 postgresql 怎么解决列名大小写问题 mysql实现列转行_字符串


需求:要求将上图中同一个用户的属性保存在一条记录中,如下图:

MySQL转 postgresql 怎么解决列名大小写问题 mysql实现列转行_SQL_02


MySqL实现代码:

SELECT owner,  
   MAX(CASE WHEN a.key = 'name' THEN a.val ELSE NULL END) as 姓名,  
   MAX(CASE WHEN a.key = 'age'  THEN a.val ELSE NULL END) as 年龄  
FROM `table1` as a  
WHERE owner = 1 group by owner

注意:实现代码中用到了MAX函数,这个很重要,它可以返回多个字符串中ASCII最大的字符串,不过使用中要注意,它可能改变返回的数据类型(String->BLOB)。另外,MAX也可以用GROUP_CONCAT函数代替,它是想同一列的记录连接起来。

二,行转列

即将原本同一列下多行的不同内容作为多个字段,输出对应内容。

原表中的数据:

MySQL转 postgresql 怎么解决列名大小写问题 mysql实现列转行_SQL_03


转换后的结果:

MySQL转 postgresql 怎么解决列名大小写问题 mysql实现列转行_字段_04


1、使用case…when…then 进行行转列

SELECT userid,
  SUM(CASE `subject` WHEN '语文' THEN score ELSE 0 END) as '语文',
  SUM(CASE `subject` WHEN '数学' THEN score ELSE 0 END) as '数学',
  SUM(CASE `subject` WHEN '英语' THEN score ELSE 0 END) as '英语',
  SUM(CASE `subject` WHEN '政治' THEN score ELSE 0 END) as '政治' 
FROM tb_score 
GROUP BY userid

2、使用IF() 进行行转列:

SELECT userid,
	SUM(IF(`subject`='语文',score,0)) as '语文',
	SUM(IF(`subject`='数学',score,0)) as '数学',
	SUM(IF(`subject`='英语',score,0)) as '英语',
	SUM(IF(`subject`='政治',score,0)) as '政治' 
FROM tb_score 
GROUP BY userid

注意点:

(1) SUM() 是为了能够使用GROUP BY根据userid进行分组,因为每一个userid对应的subject="语文"的记录只有一条,所以SUM() 的值就等于对应那一条记录的score的值。

假如userid =‘001’ and subject=‘语文’ 的记录有两条,则此时SUM() 的值将会是这两条记录的和,同理,使用Max()的值将会是这两条记录里面值最大的一个。但是正常情况下,一个user对应一个subject只有一个分数,因此可以使用SUM()、MAX()、MIN()、AVG()等聚合函数都可以达到行转列的效果。

(2)IF(subject=‘语文’,score,0) 作为条件,即对所有subject='语文’的记录的score字段进行SUM()、MAX()、MIN()、AVG()操作,如果score没有值则默认为0。

3、利用SUM(IF()) 生成列 + WITH ROLLUP 生成汇总行,并利用 IFNULL将汇总行标题显示为Total

SELECT IFNULL(userid,'total') AS userid,
	SUM(IF(`subject`='语文',score,0)) AS 语文,
	SUM(IF(`subject`='数学',score,0)) AS 数学,
	SUM(IF(`subject`='英语',score,0)) AS 英语,
	SUM(IF(`subject`='政治',score,0)) AS 政治,
	SUM(IF(`subject`='total',score,0)) AS total
FROM(
    SELECT userid,IFNULL(`subject`,'total') AS `subject`,SUM(score) AS score
    FROM tb_score
    GROUP BY userid,`subject`
    WITH ROLLUP
    HAVING userid IS NOT NULL
)AS A 
GROUP BY userid
WITH ROLLUP;

运行结果:

MySQL转 postgresql 怎么解决列名大小写问题 mysql实现列转行_SQL_05

4、利用SUM(IF()) 生成列 + UNION 生成汇总行,并利用 IFNULL将汇总行标题显示为 Total

SELECT userid,
	SUM(IF(`subject`='语文',score,0)) AS 语文,
	SUM(IF(`subject`='数学',score,0)) AS 数学,
	SUM(IF(`subject`='英语',score,0)) AS 英语,
	SUM(IF(`subject`='政治',score,0)) AS 政治,
	SUM(score) AS TOTAL 
FROM tb_score
GROUP BY userid
UNION
	SELECT 'TOTAL',SUM(IF(`subject`='语文',score,0)) AS 语文,
		SUM(IF(`subject`='数学',score,0)) AS 数学,
		SUM(IF(`subject`='英语',score,0)) AS 英语,
		SUM(IF(`subject`='政治',score,0)) AS 政治,
		SUM(score) 
	FROM tb_score

5、利用SUM(IF()) 生成列,直接生成结果不再利用子查询

SELECT IFNULL(userid,'TOTAL') AS userid,
	SUM(IF(`subject`='语文',score,0)) AS 语文,
	SUM(IF(`subject`='数学',score,0)) AS 数学,
	SUM(IF(`subject`='英语',score,0)) AS 英语,
	SUM(IF(`subject`='政治',score,0)) AS 政治,
	SUM(score) AS TOTAL 
FROM tb_score
GROUP BY userid WITH ROLLUP;

6、动态,适用于列不确定情况

SET @EE='';
select @EE :=CONCAT(@EE,'sum(if(subject= \'',subject,'\',score,0)) as ',subject, ',') AS aa FROM (SELECT DISTINCT subject FROM tb_score) A ;

SET @QQ = CONCAT('select ifnull(userid,\'TOTAL\')as userid,',@EE,' sum(score) as TOTAL from tb_score group by userid WITH ROLLUP');
-- SELECT @QQ;

PREPARE stmt FROM @QQ;
EXECUTE stmt;
DEALLOCATE PREPARE stmt;

运行结果:

MySQL转 postgresql 怎么解决列名大小写问题 mysql实现列转行_行转列_06


MySQL转 postgresql 怎么解决列名大小写问题 mysql实现列转行_行转列_07


7、合并字段显示:利用group_concat()

SELECT userid,GROUP_CONCAT(`subject`,":",score)AS 成绩 FROM tb_score
GROUP BY userid

运行结果:

MySQL转 postgresql 怎么解决列名大小写问题 mysql实现列转行_行转列_08

group_concat(),手册上说明:该函数返回带有来自一个组的连接的非NULL值的字符串结果。
比较抽象,难以理解。通俗点理解,其实是这样的:group_concat()会计算哪些行属于同一组,将属于同一组的列显示出来。要返回哪些列,由函数参数(就是字段名)决定。分组必须有个标准,就是根据group by指定的列进行分组。

结论:groupconcat()函数可以很好的建属于同一分组的多个行转化为一个列。

参考链接:https://blog.51cto.com/laok8/2048603  =