SQL中Case的使用方法
Case具有兩種格式。簡單Case函數(shù)和Case搜索函數(shù)。
這兩種方式,可以實現(xiàn)相同的功能。簡單Case函數(shù)的寫法相對比較簡潔,但是和Case搜索函數(shù)相比,功能方面會有些限制,比如寫判斷式。
還有一個需要注意的問題,Case函數(shù)只返回第一個符合條件的值,剩下的Case部分將會被自動忽略。
下面我們來看一下,使用Case函數(shù)都能做些什么事情。
一,已知數(shù)據(jù)按照另外一種方式進行分組,分析。
有如下數(shù)據(jù):(為了看得更清楚,我并沒有使用國家代碼,而是直接用國家名作為Primary Key)
根據(jù)這個國家人口數(shù)據(jù),統(tǒng)計亞洲和北美洲的人口數(shù)量。應(yīng)該得到下面這個結(jié)果。
想要解決這個問題,你會怎么做?生成一個帶有洲Code的View,是一個解決方法,但是這樣很難動態(tài)的改變統(tǒng)計的方式。
如果使用Case函數(shù),SQL代碼如下:
同樣的,我們也可以用這個方法來判斷工資的等級,并統(tǒng)計每一等級的人數(shù)。SQL代碼如下;
二,用一個SQL語句完成不同條件的分組。
有如下數(shù)據(jù)
按照國家和性別進行分組,得出結(jié)果如下
普通情況下,用UNION也可以實現(xiàn)用一條語句進行查詢。但是那樣增加消耗(兩個Select部分),而且SQL語句會比較長。
下面是一個是用Case函數(shù)來完成這個功能的例子
這樣我們使用Select,完成對二維表的輸出形式,充分顯示了Case函數(shù)的強大。
三,在Check中使用Case函數(shù)。
在Check中使用Case函數(shù)在很多情況下都是非常不錯的解決方法??赡苡泻芏嗳烁揪筒挥肅heck,那么我建議你在看過下面的例子之后也嘗試一下在SQL中使用Check。
下面我們來舉個例子
公司A,這個公司有個規(guī)定,女職員的工資必須高于1000塊。如果用Check和Case來表現(xiàn)的話,如下所示
如果單純使用Check,如下所示
女職員的條件倒是符合了,男職員就無法輸入了。
--簡單Case函數(shù)CASE sexWHEN ‘1‘ THEN ‘男‘WHEN ‘2‘ THEN ‘女‘ELSE ‘其他‘ END--Case搜索函數(shù)CASE WHEN sex = ‘1‘ THEN ‘男‘WHEN sex = ‘2‘ THEN ‘女‘ELSE ‘其他‘ END
這兩種方式,可以實現(xiàn)相同的功能。簡單Case函數(shù)的寫法相對比較簡潔,但是和Case搜索函數(shù)相比,功能方面會有些限制,比如寫判斷式。
還有一個需要注意的問題,Case函數(shù)只返回第一個符合條件的值,剩下的Case部分將會被自動忽略。
--比如說,下面這段SQL,你永遠無法得到“第二類”這個結(jié)果CASE WHEN col_1 IN ( ‘a(chǎn)‘, ‘b‘) THEN ‘第一類‘WHEN col_1 IN (‘a(chǎn)‘) THEN ‘第二類‘ELSE‘其他‘ END
下面我們來看一下,使用Case函數(shù)都能做些什么事情。
一,已知數(shù)據(jù)按照另外一種方式進行分組,分析。
有如下數(shù)據(jù):(為了看得更清楚,我并沒有使用國家代碼,而是直接用國家名作為Primary Key)
國家(country) | 人口(population) |
中國 | 600 |
美國 | 100 |
加拿大 | 100 |
英國 | 200 |
法國 | 300 |
日本 | 250 |
德國 | 200 |
墨西哥 | 50 |
印度 | 250 |
根據(jù)這個國家人口數(shù)據(jù),統(tǒng)計亞洲和北美洲的人口數(shù)量。應(yīng)該得到下面這個結(jié)果。
洲 | 人口 |
亞洲 | 1100 |
北美洲 | 250 |
其他 | 700 |
想要解決這個問題,你會怎么做?生成一個帶有洲Code的View,是一個解決方法,但是這樣很難動態(tài)的改變統(tǒng)計的方式。
如果使用Case函數(shù),SQL代碼如下:
SELECT SUM(population),CASE countryWHEN ‘中國‘ THEN ‘亞洲‘WHEN ‘印度‘ THEN ‘亞洲‘WHEN ‘日本‘ THEN ‘亞洲‘WHEN ‘美國‘ THEN ‘北美洲‘WHEN ‘加拿大‘ THEN ‘北美洲‘WHEN ‘墨西哥‘ THEN ‘北美洲‘ELSE ‘其他‘ ENDFROM Table_AGROUP BY CASE countryWHEN ‘中國‘ THEN ‘亞洲‘WHEN ‘印度‘ THEN ‘亞洲‘WHEN ‘日本‘ THEN ‘亞洲‘WHEN ‘美國‘ THEN ‘北美洲‘WHEN ‘加拿大‘ THEN ‘北美洲‘WHEN ‘墨西哥‘ THEN ‘北美洲‘ELSE ‘其他‘ END;
同樣的,我們也可以用這個方法來判斷工資的等級,并統(tǒng)計每一等級的人數(shù)。SQL代碼如下;
SELECTCASE WHEN salary <= 500 THEN ‘1‘WHEN salary > 500 AND salary <= 600 THEN ‘2‘WHEN salary > 600 AND salary <= 800 THEN ‘3‘WHEN salary > 800 AND salary <= 1000 THEN ‘4‘ELSE NULL END salary_class,COUNT(*)FROM Table_AGROUP BYCASE WHEN salary <= 500 THEN ‘1‘WHEN salary > 500 AND salary <= 600 THEN ‘2‘WHEN salary > 600 AND salary <= 800 THEN ‘3‘WHEN salary > 800 AND salary <= 1000 THEN ‘4‘ELSE NULL END;
二,用一個SQL語句完成不同條件的分組。
有如下數(shù)據(jù)
國家(country) | 性別(sex) | 人口(population) |
中國 | 1 | 340 |
中國 | 2 | 260 |
美國 | 1 | 45 |
美國 | 2 | 55 |
加拿大 | 1 | 51 |
加拿大 | 2 | 49 |
英國 | 1 | 40 |
英國 | 2 | 60 |
按照國家和性別進行分組,得出結(jié)果如下
國家 | 男 | 女 |
中國 | 340 | 260 |
美國 | 45 | 55 |
加拿大 | 51 | 49 |
英國 | 40 | 60 |
普通情況下,用UNION也可以實現(xiàn)用一條語句進行查詢。但是那樣增加消耗(兩個Select部分),而且SQL語句會比較長。
下面是一個是用Case函數(shù)來完成這個功能的例子
SELECT country,SUM( CASE WHEN sex = ‘1‘ THENpopulation ELSE 0 END), --男性人口SUM( CASE WHEN sex = ‘2‘ THENpopulation ELSE 0 END) --女性人口FROM Table_AGROUP BY country;
這樣我們使用Select,完成對二維表的輸出形式,充分顯示了Case函數(shù)的強大。
三,在Check中使用Case函數(shù)。
在Check中使用Case函數(shù)在很多情況下都是非常不錯的解決方法??赡苡泻芏嗳烁揪筒挥肅heck,那么我建議你在看過下面的例子之后也嘗試一下在SQL中使用Check。
下面我們來舉個例子
公司A,這個公司有個規(guī)定,女職員的工資必須高于1000塊。如果用Check和Case來表現(xiàn)的話,如下所示
CONSTRAINT check_salary CHECK( CASE WHEN sex = ‘2‘THEN CASE WHEN salary > 1000THEN 1 ELSE 0 ENDELSE 1 END = 1 )
如果單純使用Check,如下所示
CONSTRAINT check_salary CHECK( sex = ‘2‘ AND salary > 1000 )
女職員的條件倒是符合了,男職員就無法輸入了。
四,根據(jù)條件有選擇的UPDATE。
例,有如下更新條件
但是事情沒有想象得那么簡單,假設(shè)有個人工資5000塊。首先,按照條件1,工資減少10%,變成工資4500。接下來運行第二個SQL時候,因為這個人的工資是4500在2000到4600的范圍之內(nèi), 需增加15%,最后這個人的工資結(jié)果是5175,不但沒有減少,反而增加了。如果要是反過來執(zhí)行,那么工資4600的人相反會變成減少工資。暫且不管這個規(guī)章是多么荒誕,如果想要一個SQL 語句實現(xiàn)這個功能的話,我們需要用到Case函數(shù)。代碼如下:
這里要注意一點,最后一行的ELSE salary是必需的,要是沒有這行,不符合這兩個條件的人的工資將會被寫成NUll,那可就大事不妙了。在Case函數(shù)中Else部分的默認(rèn)值是NULL,這點是需要注意的地方。
這種方法還可以在很多地方使用,比如說變更主鍵這種累活。
一般情況下,要想把兩條數(shù)據(jù)的Primary key,a和b交換,需要經(jīng)過臨時存儲,拷貝,讀回數(shù)據(jù)的三個過程,要是使用Case函數(shù)的話,一切都變得簡單多了。
假設(shè)有如上數(shù)據(jù),需要把主鍵
同樣的也可以交換兩個Unique key。需要注意的是,如果有需要交換主鍵的情況發(fā)生,多半是當(dāng)初對這個表的設(shè)計進行得不夠到位,建議檢查表的設(shè)計是否妥當(dāng)。
五,兩個表數(shù)據(jù)是否一致的檢查。
Case函數(shù)不同于DECODE函數(shù)。在Case函數(shù)中,可以使用BETWEEN,LIKE,IS NULL,IN,EXISTS等等。比如說使用IN,EXISTS,可以進行子查詢,從而 實現(xiàn)更多的功能。
下面具個例子來說明,有兩個表,tbl_A,tbl_B,兩個表中都有keyCol列。現(xiàn)在我們對兩個表進行比較,tbl_A中的keyCol列的數(shù)據(jù)如果在tbl_B的keyCol列的數(shù)據(jù)中可以找到, 返回結(jié)果‘Matched‘,如果沒有找到,返回結(jié)果‘Unmatched‘。
要實現(xiàn)下面這個功能,可以使用下面兩條語句
使用IN和EXISTS的結(jié)果是相同的。也可以使用NOT IN和NOT EXISTS,但是這個時候要注意NULL的情況。
六,在Case函數(shù)中使用合計函數(shù)
假設(shè)有下面一個表
有的學(xué)生選擇了同時修幾門課程(100,200)也有的學(xué)生只選擇了一門課程(300,400,500)。選修多門課程的學(xué)生,要選擇一門課程作為主修,主修flag里面寫入 Y。只選擇一門課程的學(xué)生,主修flag為N(實際上要是寫入Y的話,就沒有下面的麻煩事了,為了舉例子,還請多多包含)。
現(xiàn)在我們要按照下面兩個條件對這個表進行查詢
簡單的想法就是,執(zhí)行兩條不同的SQL語句進行查詢。
條件1
執(zhí)行結(jié)果1
條件2
執(zhí)行結(jié)果2
如果使用Case函數(shù),我們只要一條SQL語句就可以解決問題,具體如下所示
運行結(jié)果
通過在Case函數(shù)中嵌套Case函數(shù),在合計函數(shù)中使用Case函數(shù)等方法,我們可以輕松的解決這個問題。使用Case函數(shù)給我們帶來了更大的自由度。
最后提醒一下使用Case函數(shù)的新手注意不要犯下面的錯誤
在這個語句中When Null這一行總是返回unknown,所以永遠不會出現(xiàn)Wrong的情況。因為這句可以替換成WHEN col_1 = NULL,這是一個錯誤的用法,這個時候我們應(yīng)該選擇用WHEN col_1 IS NULL。
例,有如下更新條件
- 工資5000以上的職員,工資減少10%
- 工資在2000到4600之間的職員,工資增加15%
--條件1UPDATE PersonnelSET salary = salary * 0.9WHERE salary >= 5000;--條件2UPDATE PersonnelSET salary = salary * 1.15WHERE salary >= 2000 AND salary < 4600;
但是事情沒有想象得那么簡單,假設(shè)有個人工資5000塊。首先,按照條件1,工資減少10%,變成工資4500。接下來運行第二個SQL時候,因為這個人的工資是4500在2000到4600的范圍之內(nèi), 需增加15%,最后這個人的工資結(jié)果是5175,不但沒有減少,反而增加了。如果要是反過來執(zhí)行,那么工資4600的人相反會變成減少工資。暫且不管這個規(guī)章是多么荒誕,如果想要一個SQL 語句實現(xiàn)這個功能的話,我們需要用到Case函數(shù)。代碼如下:
UPDATE PersonnelSET salary = CASE WHEN salary >= 5000 THEN salary * 0.9WHEN salary >= 2000 AND salary < 4600THEN salary * 1.15ELSE salary END;
這里要注意一點,最后一行的ELSE salary是必需的,要是沒有這行,不符合這兩個條件的人的工資將會被寫成NUll,那可就大事不妙了。在Case函數(shù)中Else部分的默認(rèn)值是NULL,這點是需要注意的地方。
這種方法還可以在很多地方使用,比如說變更主鍵這種累活。
一般情況下,要想把兩條數(shù)據(jù)的Primary key,a和b交換,需要經(jīng)過臨時存儲,拷貝,讀回數(shù)據(jù)的三個過程,要是使用Case函數(shù)的話,一切都變得簡單多了。
p_key | col_1 | col_2 |
a | 1 | 張三 |
b | 2 | 李四 |
c | 3 | 王五 |
假設(shè)有如上數(shù)據(jù),需要把主鍵
a
和b
相互交換。用Case函數(shù)來實現(xiàn)的話,代碼如下 UPDATE SomeTableSET p_key = CASE WHEN p_key = ‘a(chǎn)‘THEN ‘b‘WHEN p_key = ‘b‘THEN ‘a(chǎn)‘ELSE p_key ENDWHERE p_key IN (‘a(chǎn)‘, ‘b‘);
同樣的也可以交換兩個Unique key。需要注意的是,如果有需要交換主鍵的情況發(fā)生,多半是當(dāng)初對這個表的設(shè)計進行得不夠到位,建議檢查表的設(shè)計是否妥當(dāng)。
五,兩個表數(shù)據(jù)是否一致的檢查。
Case函數(shù)不同于DECODE函數(shù)。在Case函數(shù)中,可以使用BETWEEN,LIKE,IS NULL,IN,EXISTS等等。比如說使用IN,EXISTS,可以進行子查詢,從而 實現(xiàn)更多的功能。
下面具個例子來說明,有兩個表,tbl_A,tbl_B,兩個表中都有keyCol列。現(xiàn)在我們對兩個表進行比較,tbl_A中的keyCol列的數(shù)據(jù)如果在tbl_B的keyCol列的數(shù)據(jù)中可以找到, 返回結(jié)果‘Matched‘,如果沒有找到,返回結(jié)果‘Unmatched‘。
要實現(xiàn)下面這個功能,可以使用下面兩條語句
--使用IN的時候SELECT keyCol,CASE WHEN keyCol IN ( SELECT keyCol FROM tbl_B )THEN ‘Matched‘ELSE ‘Unmatched‘ END LabelFROM tbl_A;--使用EXISTS的時候SELECT keyCol,CASE WHEN EXISTS ( SELECT * FROM tbl_BWHERE tbl_A.keyCol = tbl_B.keyCol )THEN ‘Matched‘ELSE ‘Unmatched‘ END LabelFROM tbl_A;
使用IN和EXISTS的結(jié)果是相同的。也可以使用NOT IN和NOT EXISTS,但是這個時候要注意NULL的情況。
六,在Case函數(shù)中使用合計函數(shù)
假設(shè)有下面一個表
學(xué)號(std_id) | 課程ID(class_id) | 課程名(class_name) | 主修flag(main_class_flg) |
100 | 1 | 經(jīng)濟學(xué) | Y |
100 | 2 | 歷史學(xué) | N |
200 | 2 | 歷史學(xué) | N |
200 | 3 | 考古學(xué) | Y |
200 | 4 | 計算機 | N |
300 | 4 | 計算機 | N |
400 | 5 | 化學(xué) | N |
500 | 6 | 數(shù)學(xué) | N |
有的學(xué)生選擇了同時修幾門課程(100,200)也有的學(xué)生只選擇了一門課程(300,400,500)。選修多門課程的學(xué)生,要選擇一門課程作為主修,主修flag里面寫入 Y。只選擇一門課程的學(xué)生,主修flag為N(實際上要是寫入Y的話,就沒有下面的麻煩事了,為了舉例子,還請多多包含)。
現(xiàn)在我們要按照下面兩個條件對這個表進行查詢
- 只選修一門課程的人,返回那門課程的ID
- 選修多門課程的人,返回所選的主課程ID
簡單的想法就是,執(zhí)行兩條不同的SQL語句進行查詢。
條件1
--條件1:只選擇了一門課程的學(xué)生SELECT std_id, MAX(class_id) AS main_classFROM StudentclassGROUP BY std_idHAVING COUNT(*) = 1;
執(zhí)行結(jié)果1
STD_ID MAIN_class------ ----------300 4400 5500 6
條件2
--條件2:選擇多門課程的學(xué)生SELECT std_id, class_id AS main_classFROM StudentclassWHERE main_class_flg = ‘Y‘ ;
執(zhí)行結(jié)果2
STD_ID MAIN_class------ ----------100 1200 3
如果使用Case函數(shù),我們只要一條SQL語句就可以解決問題,具體如下所示
SELECT std_id,CASE WHEN COUNT(*) = 1 --只選擇一門課程的學(xué)生的情況THEN MAX(class_id)ELSE MAX(CASE WHEN main_class_flg = ‘Y‘THEN class_idELSE NULL END)END AS main_classFROM StudentclassGROUP BY std_id;
運行結(jié)果
STD_ID MAIN_class------ ----------100 1200 3300 4400 5500 6
通過在Case函數(shù)中嵌套Case函數(shù),在合計函數(shù)中使用Case函數(shù)等方法,我們可以輕松的解決這個問題。使用Case函數(shù)給我們帶來了更大的自由度。
最后提醒一下使用Case函數(shù)的新手注意不要犯下面的錯誤
CASE col_1WHEN 1 THEN ‘Right‘WHEN NULL THEN ‘Wrong‘END
在這個語句中When Null這一行總是返回unknown,所以永遠不會出現(xiàn)Wrong的情況。因為這句可以替換成WHEN col_1 = NULL,這是一個錯誤的用法,這個時候我們應(yīng)該選擇用WHEN col_1 IS NULL。