MySQL列轉行、行轉列
標簽: MySQL
準備數據
創建表:
DROP TABLE IF EXISTS tb_score;
CREATE TABLE tb_score(
id INT(11) NOT NULL auto_increment,
userid VARCHAR(20) NOT NULL COMMENT '用戶id',
subject VARCHAR(20) COMMENT '科目',
score DOUBLE COMMENT '成績',
PRIMARY KEY(id)
)ENGINE = INNODB DEFAULT CHARSET = utf8;
插入數據:
INSERT INTO tb_score(userid,subject,score) VALUES ('001','語文',90);
INSERT INTO tb_score(userid,subject,score) VALUES ('001','數學',92);
INSERT INTO tb_score(userid,subject,score) VALUES ('001','英語',80);
INSERT INTO tb_score(userid,subject,score) VALUES ('002','語文',88);
INSERT INTO tb_score(userid,subject,score) VALUES ('002','數學',90);
INSERT INTO tb_score(userid,subject,score) VALUES ('002','英語',75.5);
INSERT INTO tb_score(userid,subject,score) VALUES ('003','語文',70);
INSERT INTO tb_score(userid,subject,score) VALUES ('003','數學',85);
INSERT INTO tb_score(userid,subject,score) VALUES ('003','英語',90);
INSERT INTO tb_score(userid,subject,score) VALUES ('003','政治',82);
SELECT * FROM tb_score
WITH ROLLUP 的用法
1.WITH ROLLUP:在group分組字段的基礎上再進行統計數據。
SELECT userid,IFNULL(`subject`,'total') AS `subject`,SUM(score) AS score
FROM tb_score
GROUP BY userid,`subject`
WITH ROLLUP
上面的結果第是為每個userid進行的統計,最后的一列是對所有的total進行的統計
MySQL行轉列
要將上面的數據轉換成為下面的格式
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 生成匯總行
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列轉行
創建表:
CREATE TABLE tb_score1(
id INT(11) NOT NULL auto_increment,
userid VARCHAR(20) NOT NULL COMMENT '用戶id',
cn_score DOUBLE COMMENT '語文成績',
math_score DOUBLE COMMENT '數學成績',
en_score DOUBLE COMMENT '英語成績',
po_score DOUBLE COMMENT '政治成績',
PRIMARY KEY(id)
)ENGINE = INNODB DEFAULT CHARSET = utf8;
插入數據:
INSERT INTO tb_score1(userid,cn_score,math_score,en_score,po_score) VALUES ('001',90,92,80,0);
INSERT INTO tb_score1(userid,cn_score,math_score,en_score,po_score) VALUES ('002',88,90,75.5,0);
INSERT INTO tb_score1(userid,cn_score,math_score,en_score,po_score) VALUES ('003',70,85,90,82);
結果如下:
本質是將userid的每個科目分數分散成一條記錄顯示出來。
SELECT userid,'語文' AS course,cn_score AS score FROM tb_score1
UNION ALL
SELECT userid,'數學' AS course,math_score AS score FROM tb_score1
UNION ALL
SELECT userid,'英語' AS course,en_score AS score FROM tb_score1
UNION ALL
SELECT userid,'政治' AS course,po_score AS score FROM tb_score1
ORDER BY userid
UNION與UNION ALL的區別:
1.對重復結果的處理:UNION會去掉重復記錄,UNION ALL不會;
2.對排序的處理:UNION會排序,UNION ALL只是簡單地將兩個結果集合并;
3.效率方面的區別:因為UNION 會做去重和排序處理,因此效率比UNION ALL慢很多;
參考自:https://www.cnblogs.com/yinjw/p/11767282.html
智能推薦
Python基本數據類型
Python基本數據類型 一、數字型 (一)、整數型 (二)、浮點型 (三)、布爾型 (四)、復數型 二、字符串 三、列表 四、元組 五、集合 六、字典 一、數字型 (一)、整數型 1、整型:int 在數字中,正整數、0、負整數都稱為整型。 例: 運行結果: 2、二進制整型 也可用二進制表示整型,print自動轉換為十進制。 例: 運行結果: (二)、浮點型 1、浮點型:float 含有小數點的數...
安卓轉場動畫
一、系統自帶動畫效果 假如有 A 和 B 兩個 Activity, 從 A 進入 B,調用 startActivity(Intent(this,AnimCutOutActivity::class.java)) overridePendingTransition(android.R.anim.fade_in,android.R.anim.fade_out) 第一個動畫android.R.anim.f...
瀏覽器與服務端的通信01
在pom.xml中導入兩個jar包:jaxen1.1.4和dom4j1.6.1 WebServer類: ClientHandler類: HttpContent類: ServerContent類: HttpRequest類: HttpResponse類: server配置文件:...
用Flutter 寫一個簡單頁面
奇技指南 本文適用于對Flutter 感興趣的初學者。筆者會帶大家寫一個簡單頁面,使大家對Flutter有一個簡單地了解。 Flutter 簡介 Flutter是Google的UI工具包,可使用一套代碼庫,為移動,Web和桌面構建漂亮的原生編譯的應用程序。 Flutter實現了一個自繪引擎,使用自身的布局、繪制系統,繪制界面。 Flutter 框架架構圖 Flutter框架圖簡單說明:...
猜你喜歡
山東大學機器學習(實驗五解讀)——SVM
1.SVM (1)這里我選擇懲罰系數C=1C=1C=1做實驗,不同的懲罰系數CCC可能導致結果不同。 核函數kernel.m 訓練函數svmTrain.m 預測函數predict1.m 主函數part1.m 做出圖像如下 training_1.txt 和 test_1.txt training_2.txt 和 test_2.txt (2)測試數據的預測結果如下,可以由上面第(1)問中的代碼獲得 t...
C語言重構【11】盛最多水的容器
文章目錄 所有題目源代碼:[Git地址](https://github.com/ch98road/leetcode) 題目 方案: 復雜度計算 所有題目源代碼:Git地址 題目 給你 n 個非負整數 a1,a2,…,an,每個數代表坐標中的一個點 (i, ai) 。在坐標內畫 n 條垂直線,垂直線 i 的兩個端點分別為 (i, ai) 和 (i, 0)。找出其中的兩條線,使得它們與 ...
力扣周賽 239 題解
1848. 到目標元素的最小距離 時間復雜度:O(n) 從 start 向兩端尋找 target,找到的第一個目標元素即最近的。 1849. 將字符串拆分為遞減的連續值 知識點:遞歸 時間復雜度:O(n^2) 因為兩個數字相差必須是1,所以只要前一個數字確定了,當前數字的值就確定了。 首先,枚舉第一個數字的長度,一旦長度確定了,值就確定了,后續數字的值也就確定了。然后按值嘗試分割字符串即可。 需要...