Mysql

关注公众号 jb51net

关闭
首页 > 数据库 > Mysql > MySQL行转列

一文带你解锁MySQL实现行转列的完整方法

作者:detayun

MySQL的行转列,不是简单的语法堆砌,而是对数据结构深刻理解后的重构,这篇文章主要介绍了MySQL实现行转列的完整方法,有需要的小伙伴可以了解下

在数据处理的江湖中,我们常面临这样一种尴尬的局面:数据库里的数据明明就在那里,却像是一堆散乱的拼图,无法以直观的报表形式呈现。比如,学生的成绩单,数据库里存的是“张三-语文-90”、“张三-数学-92”这样的行记录,而我们要看的却是“张三 | 语文90 | 数学92”这样的宽表。

这就是行转列(Pivot)的战场。MySQL虽然不像Oracle或SQL Server那样原生支持PIVOT关键字,但它提供了足够锋利的利器。今天,我们就来剥开行转列的核心原理,掌握这门数据炼金术。

一、 核心心法:聚合+条件判断

行转列的本质,是将“行的维度”压缩,转化为“列的维度”。实现这一魔术的核心公式只有一条:

GROUP BY + 聚合函数(SUM/MAX/MIN) + 条件判断(CASE WHEN/IF)

如果没有聚合函数,多行数据无法坍缩为一行;如果没有条件判断,数据无法精准地填充到对应的列中。

经典招式:CASE WHEN与SUM(IF)的对决

假设我们有一张成绩表 tb_score,存储了用户ID、科目和分数。我们要将其转为以用户ID为行,各科目为列的报表。

场景数据

CREATE TABLE tb_score(
    id INT AUTO_INCREMENT,
    userid VARCHAR(20),
    subject VARCHAR(20),
    score DOUBLE,
    PRIMARY KEY(id)
);
INSERT INTO tb_score(userid,subject,score) VALUES 
('001','语文',90), ('001','数学',92), ('001','英语',80),
('002','语文',88), ('002','数学',90), ('002','英语',75.5);

招式一:CASE WHEN(标准SQL,通用性强)

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 '英语'
FROM tb_score 
GROUP BY userid;

招式二:SUM(IF(...))(MySQL特色,简洁高效)

SELECT 
    userid,
    SUM(IF(subject='语文', score, 0)) AS '语文',
    SUM(IF(subject='数学', score, 0)) AS '数学',
    SUM(IF(subject='英语', score, 0)) AS '英语'
FROM tb_score 
GROUP BY userid;

高手进阶:为什么用SUM

很多人会问:明明每个用户每个科目只有一条记录,为什么不用MAXMIN

这里有一个关键细节:SUM在这里不仅是求和,更是为了配合GROUP BY进行行坍缩

二、 奇门遁甲:应对复杂场景

基础的行转列只能解决静态列的问题,面对动态列、列转行或字符串聚合,我们需要更高级的战术。

1. 字符串聚合:GROUP_CONCAT

如果不想要数值列,而是想把多行文本合并成一个字符串(比如合并标签),GROUP_CONCAT是神器。

-- 将同一用户的所有分数合并显示
SELECT userid, GROUP_CONCAT(score) AS all_scores 
FROM tb_score 
GROUP BY userid;

2. 列转行(Unpivot):UNION ALL

这是行转列的逆操作。如果表结构是 userid | 语文 | 数学 | 英语,想转回行结构:

SELECT userid, '语文' AS subject, 语文 AS score FROM tb_score_wide
UNION ALL
SELECT userid, '数学' AS subject, 数学 AS score FROM tb_score_wide
UNION ALL
SELECT userid, '英语' AS subject, 英语 AS score FROM tb_score_wide;

虽然写法繁琐,但这是SQL标准处理列转行的不二法门。

3. 动态行转列:预处理语句(Prepared Statement)

这是最考验功力的一招。当科目(列名)不确定,可能随时增加“物理”、“化学”时,写死SQL是不可能的。必须动态生成SQL语句并执行。

核心逻辑

SET @sql = NULL;
SELECT GROUP_CONCAT(
    DISTINCT CONCAT('SUM(CASE subject WHEN ''', subject, ''' THEN score ELSE 0 END) AS `', subject, '`')
) INTO @sql 
FROM tb_score;

SET @sql = CONCAT('SELECT userid, ', @sql, ' FROM tb_score GROUP BY userid');

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

这段代码能自动适应表中所有的科目,是生成动态报表的终极武器。

三、 实战锦囊:性能与优化

行转列虽然强大,但也是性能杀手。因为它需要全表扫描并进行分组排序,数据量大时极易拖慢数据库。

结语

MySQL的行转列,不是简单的语法堆砌,而是对数据结构深刻理解后的重构。从死板的CASE WHEN到灵活的动态SQL,每一种方法都对应着特定的业务痛点。

掌握它,你就掌握了将“丑数据”变为“黄金报表”的能力。在数据分析的道路上,这不仅是一门技术,更是一门艺术。现在,打开你的MySQL客户端,开始你的炼金之旅吧!

到此这篇关于一文带你解锁MySQL实现行转列的完整方法的文章就介绍到这了,更多相关MySQL行转列内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文