| 方一凡 | 70 | 90 | 59 | 200 |
| 陈哈哈 | 109 | 92 | 80 | 0 |
±-------------±-------±-------±-------±-------------+
4 rows in set (0.00 sec)
好,下面我们一起来看看SQL是如何编写的,对了,`创建表结构和导入测试数据的SQL放到文章末尾了`,自取~
---
#### 飞机票
* [一、行转列SQL写法](#SQL_53)
* [二、如果领导@你,让你在结果集中加上总数列呢?](#_91)
* [三、领导又双叒叕@你改需求](#_126)
* [四、结束语](#_214)
* [附录:创建表结构&测试数据SQL](#SQL_225)
---
## 一、行转列SQL写法
* 方法一、使用`case..when..then`进行 行转列
SELECT 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()`。
>
>
>
![在这里插入图片描述](https://img-blog.csdnimg.cn/69324c9052c041a98f2972190ebee17e.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAX-mZiOWTiOWTiA==,size_20,color_FFFFFF,t_70,g_se,x_16#pic_center)
* 方法二、使用`IF()`进行 行转列:
SELECT 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) f
rom 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方面就是以上这些内容了,有疑问可以写在评论区,哈哥会在摸鱼的时候回复你~~`
帮忙三连一下哦,比心ღ( ´・ᴗ・` )
![在这里插入图片描述](https://img-blog.csdnimg.cn/3f33a3c904e14f17aa7f9772a5ccbe0c.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAX-mZiOWTiOWTiA==,size_20,color_FFFFFF,t_70,g_se,x_16#pic_center)
## 附录:创建表结构&测试数据SQL
### 最后
给大家送一个小福利
![](https://img-blog.csdnimg.cn/img_convert/946808e60090c0c968e5d5f3f5f433f2.webp?x-oss-process=image/format,png)
附高清脑图,高清知识点讲解教程,以及一些面试真题及答案解析。送给需要的提升技术、准备面试跳槽、自身职业规划迷茫的朋友们。
![](https://img-blog.csdnimg.cn/img_convert/d37a9376a038300e1d84d07b2a5d37e1.webp?x-oss-process=image/format,png)
TiA==,size_20,color_FFFFFF,t_70,g_se,x_16#pic_center)
## 附录:创建表结构&测试数据SQL
### 最后
给大家送一个小福利
[外链图片转存中...(img-kFj0xbsC-1723360194354)]
附高清脑图,高清知识点讲解教程,以及一些面试真题及答案解析。送给需要的提升技术、准备面试跳槽、自身职业规划迷茫的朋友们。
[外链图片转存中...(img-ElTq3L6f-1723360194355)]
> **本文已被[CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】](https://docs.qq.com/doc/DSmxTbFJ1cmN1R2dB)收录**