MySQL行转列详情
本篇文章主要是结合我之前面试的各种经历和实战开发中遇到的问题解决经验整理的,希望这篇《MySQL行转列详情》对你有很大帮助!欢迎收藏,分享给更多的需要的朋友学习~
MySQL行转列,对经常处理数据的同学们来说,一定是不陌生的,甚至是印象深刻,因为它大概率困扰过你,让你为之一愣~ 但当你看到本文后,这个问题就不在是问题,及时收藏,以后谁再问你这个问题,直接甩他脸上,粘贴即用。
首先,我们看一下咱们的测试表数据和预期查询的结果:
mysql> SELECT * FROM t_gaokao_score; +----+--------------+--------------+-------+ | id | student_name | subject | score | +----+--------------+--------------+-------+ | 1 | 林磊儿 | 语文 | 148 | | 2 | 林磊儿 | 数学 | 150 | | 3 | 林磊儿 | 英语 | 147 | | 4 | 乔英子 | 语文 | 121 | | 5 | 乔英子 | 数学 | 106 | | 6 | 乔英子 | 英语 | 146 | | 7 | 方一凡 | 语文 | 70 | | 8 | 方一凡 | 数学 | 90 | | 9 | 方一凡 | 英语 | 59 | | 10 | 方一凡 | 特长加分 | 200 | | 11 | 陈哈哈 | 语文 | 109 | | 12 | 陈哈哈 | 数学 | 92 | | 13 | 陈哈哈 | 英语 | 80 | +----+--------------+--------------+-------+ 13 rows in set (0.00 sec)
看看我们行转列转完后的结果:
+--------------+--------+--------+--------+--------------+ | student_name | 语文 | 数学 | 英语 | 特长加分 | +--------------+--------+--------+--------+--------------+ | 林磊儿 | 148 | 150 | 147 | 0 | | 乔英子 | 121 | 106 | 146 | 0 | | 方一凡 | 70 | 90 | 59 | 200 | | 陈哈哈 | 109 | 92 | 80 | 0 | +--------------+--------+--------+--------+--------------+ 4 rows in set (0.00 sec)
好,下面我们一起来看看SQL是如何编写的,对了,创建表结构和导入测试数据的SQL放到文章末尾了,自取~
一、行转列SQL写法
方法一、使用case..when..then进行 行转列
ELECT student_name, 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 t_gaokao_score GROUP BY student_name;
这里如果不使用SUM()会报sql_mode=only_full_group_by相关错误,需要聚合函数和group by连用或使用distinct才可以解决。
其实,加了SUM()是为了能够使用GROUP BY根据student_name进行分组,每一个student_name对应的subject="语文"的记录毕竟只有一条,所以SUM() 的值就等于对应那一条记录的score的值。当然,也可以换成MAX()。
方法二、使用IF()进行 行转列:
ELECT student_name, 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 t_gaokao_score GROUP BY student_name;
该方法将IF(subject='语文',score,0)作为条件,通过student_name进行分组,对分组后所有subject='语文’的记录的score字段进行SUM()操作,如果score没有值则默认为0。
这种方式和case..when..then方法原理相同,相比更加简洁明了,建议使用。
二、如果领导@你,让你在结果集中加上总数列呢?
友情提示:我们工作中处理行转列数据时,尽量都把总数、平均数等加上,方便领导查阅,省得他循环BB你。
话说,你还记得上学时的成绩表是啥样的么?你一般从上往下看还是从下往上看呢?文末投票,快来给大家乐呵乐呵!
写法:利用SUM(IF()) 生成列,WITH ROLLUP 生成汇总列和行,并利用 IFNULL将汇总行标题显示为总数
SELECT IFNULL(student_name,'总数') AS student_name, 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 '总数' FROM t_gaokao_score GROUP BY student_name WITH ROLLUP;
查询结果:
+--------------+--------+--------+--------+--------------+--------+ | student_name | 语文 | 数学 | 英语 | 特长加分 | 总数 | +--------------+--------+--------+--------+--------------+--------+ | 乔英子 | 121 | 106 | 146 | 0 | 373 | | 方一凡 | 70 | 90 | 59 | 200 | 419 | | 林磊儿 | 148 | 150 | 147 | 0 | 445 | | 陈哈哈 | 113 | 116 | 80 | 0 | 309 | | 总数 | 452 | 462 | 432 | 200 | 1546 | +--------------+--------+--------+--------+--------------+--------+ 5 rows in set, 1 warning (0.00 sec)
三、领导又双叒叕@你改需求
让你把分值转化为具体内容显示(优秀、良好、普通、差),430分以上重点大学,400分以上一本,350分及以上二本,350以下搬砖,该怎么写呢?
这里我们就需要case when嵌套一下了,看着高大上,其实就是普通的嵌套而已。在第一层查出分组后的各科分数,在第二层替换成等级即可。
SELECT student_name, MAX( CASE subject WHEN '语文' THEN ( CASE WHEN score - (select avg(score) from t_gaokao_score where subject='语文') > 20 THEN '优秀' WHEN score - (select avg(score) from t_gaokao_score where subject='语文') > 10 THEN '良好' WHEN score - (select avg(score) from t_gaokao_score where subject='语文') >= 0 THEN '普通' ELSE '差' END ) END ) as '语文', MAX( CASE subject WHEN '数学' THEN ( CASE WHEN score - (select avg(score) from t_gaokao_score where subject='数学') > 20 THEN '优秀' WHEN score - (select avg(score) from t_gaokao_score where subject='数学') > 10 THEN '良好' WHEN score - (select avg(score) from t_gaokao_score where subject='数学') >= 0 THEN '普通' ELSE '差' END ) END ) as '数学', MAX( CASE subject WHEN '英语' THEN ( CASE WHEN score - (select avg(score) from t_gaokao_score where subject='英语') > 20 THEN '优秀' WHEN score - (select avg(score) from t_gaokao_score where subject='英语') > 10 THEN '良好' WHEN score - (select avg(score) from t_gaokao_score where subject='英语') >= 0 THEN '普通' ELSE '差' END ) END ) as '英语', SUM(score) as '总分', (CASE WHEN SUM(score) > 430 THEN '重点大学' WHEN SUM(score) > 400 THEN '一本' WHEN SUM(score) > 350 THEN '二本' ELSE '工地搬砖' END ) as '结果' FROM t_gaokao_score GROUP BY student_name ORDER BY SUM(score) desc;
我们来看一下输出结果:
+--------------+--------+--------+--------+--------+--------------+ | student_name | 语文 | 数学 | 英语 | 总分 | 结果 | +--------------+--------+--------+--------+--------+--------------+ | 林磊儿 | 优秀 | 优秀 | 优秀 | 445 | 重点大学 | | 方一凡 | 差 | 差 | 差 | 419 | 一本 | | 乔英子 | 普通 | 差 | 优秀 | 373 | 二本 | | 陈哈哈 | 普通 | 普通 | 差 | 309 | 工地搬砖 | +--------------+--------+--------+--------+--------+--------------+ 4 rows in set (0.00 sec)
过来人的经验来看,老实孩子最吃亏,早知道他娘的走艺体了~
四、结束语
好了,SQL方面就是以上这些内容了,有疑问可以写在评论区,哈哥会在摸鱼的时候回复你~~`
附录:创建表结构&测试数据SQL
表结构:
DROP TABLE IF EXISTS `t_gaokao_score`; CREATE TABLE `t_gaokao_score` ( `id` int(0) NOT NULL AUTO_INCREMENT, `student_name` varchar(20) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '学生姓名', `subject` varchar(20) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '科目', `score` double NULL DEFAULT NULL COMMENT '成绩', PRIMARY KEY (`id`) USING BTREE ) ENGINE = InnoDB AUTO_INCREMENT = 11 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Dynamic;
导入测试数据:
INSERT INTO `t_gaokao_score` VALUES (1, '林磊儿', '语文', 148), (2, '林磊儿', '数学', 150), (3, '林磊儿', '英语', 147), (4, '乔英子', '语文', 121), (5, '乔英子', '数学', 106), (6, '乔英子', '英语', 146), (7, '方一凡', '语文', 70), (8, '方一凡', '数学', 90), (9, '方一凡', '英语', 59), (10, '方一凡', '特长加分', 200), (11, '陈哈哈', '语文', 109), (12, '陈哈哈', '数学', 92), (13, '陈哈哈', '英语', 80);
文中关于mysql的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《MySQL行转列详情》文章吧,也可关注golang学习网公众号了解相关技术文章。
MySQL百万级数据大分页查询优化的实现
- 上一篇
- MySQL百万级数据大分页查询优化的实现
- 下一篇
- MySQL 中定位 DDL 被阻塞的问题及解决方案
-
- 舒适的火
- 这篇博文真是及时雨啊,太详细了,写的不错,mark,关注up主了!希望up主能多写数据库相关的文章。
- 2023-03-11 12:44:00
-
- 淡定的缘分
- 太全面了,码起来,感谢师傅的这篇博文,我会继续支持!
- 2023-03-08 01:30:45
-
- 典雅的歌曲
- 好细啊,码住,感谢博主的这篇文章,我会继续支持!
- 2023-03-07 08:01:56
-
- 害羞的玫瑰
- 这篇文章内容出现的刚刚好,大佬加油!
- 2023-03-04 09:52:00
-
- 过时的香菇
- 这篇博文真及时,细节满满,感谢大佬分享,已收藏,关注up主了!希望up主能多写数据库相关的文章。
- 2023-03-04 03:52:21
-
- 俊秀的画笔
- 很棒,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢博主分享技术文章!
- 2023-03-01 02:26:00
-
- 淡然的萝莉
- 很棒,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢作者大大分享技术贴!
- 2023-02-27 02:03:26
-
- 数据库 · MySQL | 2星期前 | MySQL · 慢查询 · 索引优化 · COUNT查询 · 汇总表 · 联合索引 覆盖索引 汇总表 MySQL COUNT慢 COUNT(*)优化
- MySQL COUNT(*) 总数查询变慢怎么办:从扫描行数到汇总表的完整治理流程
- 329浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ljg-skills
- ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
- 3202次使用
-
- MELO音乐
- MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
- 2954次使用
-
- UniScribe
- UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
- 2909次使用
-
- 剧云
- 剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
- 3112次使用
-
- 万象有声
- 万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
- 3069次使用
-
- 学习mysql 如何行转列与列传行
- 2023-01-07 481浏览

