CASE WHEN 及 SELECT CASE WHEN的用法、SUM函式中的數字引數(轉)
種方式,可以實現相同的功能。簡單Case函式的寫法相對比較簡潔,但是和Case搜尋函式相比,功能方面會有些限制,比如寫判斷式。還有一個需要注意的問題,Case函式只返回第一個符合條件的值,剩下的Case部分將會被自動忽略。 --比如說,下面這段SQL,你永遠無法得到“第二類”這個結果 CASE
下面我們來看一下,使用Case函式都能做些什麼事情。
一,已知資料按照另外一種方式進行分組,分析。
有如下資料:(為了看得更清楚,我並沒有使用國家程式碼,而是直接用國家名作為Primary Key) 根據這個國家人口資料,統計亞洲和北美洲的人口數量。應該得到下面這個結果。
解決這個問題,你會怎麼做?生成一個帶有洲Code的View是一個解決方法,但是這樣很難動態的改變統計的方式。如果使用Case函式,SQL程式碼如下: SELECT
同樣的,我們也可以用這個方法來判斷工資的等級,並統計每一等級的人數。SQL程式碼如下: SELECT CASE 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_A GROUP BY CASE 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語句完成不同條件的分組。
有如下資料
按照國家和性別進行分組,得出結果如下
普通情況下,用UNION也可以實現用一條語句進行查詢。但是那樣增加消耗(兩個Select部分),而且SQL語句會比較長。下面是一個是用Case函式來完成這個功能的例子
SELECT country, SUM( CASE WHEN sex = '1' THEN population ELSE 0 END), --男性人口 SUM( CASE WHEN sex = '2' THEN population ELSE 0 END) --女性人口 FROM Table_A GROUP BY country;
這樣我們使用Select,完成對二維表的輸出形式,充分顯示了Case函式的強大。
三,在Check中使用Case函式。
在Check中使用Case函式在很多情況下都是非常不錯的解決方法。可能有很多人根本就不用Check,那麼我建議你在看過下面的例子之後也嘗試一下在SQL中使用Check。
下面我們來舉個例子
公司A,這個公司有個規定,女職員的工資必須高於1000塊。如果用Check和Case來表現的話,如下所示
CONSTRAINT check_salary CHECK ( CASE WHEN sex = '2' THEN CASE WHEN salary > 1000 THEN1 ELSE 0 END ELSE 1 END = 1 )
如果單純使用Check: CONSTRAINT check_salary CHECK ( sex = '2' AND salary > 1000 ) 女職員的條件倒是符合了,男職員就無法輸入了。
四,根據條件有選擇的UPDATE。例,有如下更新條件 1.工資5000以上的職員,工資減少10%
2.工資在2000到4600之間的職員,工資增加15%
很容易考慮的是選擇執行兩次UPDATE語句,如下所示 --條件1 UPDATE Personnel SET salary = salary * 0.9 WHERE salary >= 5000; --條件2 UPDATE Personnel SET salary = salary * 1.15 WHERE salary >= 2000 AND salary < 4600;但是事情沒有想象得那麼簡單,假設有個人工資5000塊。首先,按照條件1,工資減少10%,變成工資4500。接下來執行第二個SQL時候,因為這個人的工資是4500在2000到4600的範圍之內,需增加15%,最後這個人的工資結果是5175,不但沒有減少,反而增加了。如果要是反過來執行,那麼工資4600的人相反會變成減少工資。暫且不管這個規章是多麼荒誕,如果想要一個SQL 語句實現這個功能的話,我們需要用到Case函式。程式碼如下: UPDATE Personnel SET salary = CASE WHEN salary >= 5000 THEN salary * 0.9 WHEN salary >= 2000 AND salary < 4600 THEN salary * 1.15 ELSE salary END;
這裡要注意一點,最後一行的ELSE salary是必需的,要是沒有這行,不符合這兩個條件的人的工資將會被寫成NUll,那可就大事不妙了。在Case函式中Else部分的預設值是NULL,這點是需要注意的地方。
這種方法還可以在很多地方使用,比如說變更主鍵這種累活。
一般情況下,要想把兩條資料的Primary key,a和b交換,需要經過臨時儲存,拷貝,讀回資料的三個過程,要是使用Case函式的話,一切都變得簡單多了。
p_key | col_1 | col_2 |
a | 1 | 張三 |
b | 2 | 李四 |
c | 3 | 王五 |
a
和b
相互交換。用Case函式來實現的話,程式碼如下
UPDATE SomeTable
SET p_key = CASE WHEN p_key
= 'a' THEN 'b'
WHEN p_key = 'b' THEN 'a' ELSE p_key END
WHERE p_key IN ('a', 'b');
同樣的也可以交換兩個Unique key。需要注意的是,如果有需要交換主鍵的情況發生,多半是當初對這個表的設計進行得不夠到位,建議檢查表的設計是否妥當。
五,兩個表資料是否一致的檢查。
Case函式不同於DECODE函式。在Case函式中,可以使用BETWEEN,LIKE,IS NULL,IN,EXISTS等等。比如說使用IN,EXISTS,可以進行子查詢,從而 實現更多的功能。
下面具個例子來說明,有兩個表,tbl_A,tbl_B,兩個表中都有keyCol列。現在我們對兩個表進行比較,tbl_A中的keyCol列的資料如果在tbl_B的keyCol列的資料中可以找到,返回結果'Matched',如果沒有找到,返回結果'Unmatched'。
要實現下面這個功能,可以使用下面兩條語句 --使用IN的時候 SELECT keyCol, CASE WHEN keyCol IN ( SELECT keyCol FROM tbl_B ) THEN 'Matched' ELSE 'Unmatched' END Label FROM tbl_A; --使用EXISTS的時候 SELECT keyCol, CASE WHEN EXISTS ( SELECT * FROM tbl_B WHERE tbl_A.keyCol = tbl_B.keyCol ) THEN'Matched' ELSE 'Unmatched' END Label FROM tbl_A; 使用IN和EXISTS的結果是相同的。也可以使用NOT IN和NOT EXISTS,但是這個時候要注意NULL的情況。
六,在Case函式中使用合計函式
假設有下面一個表
學號(std_id) | 課程ID(class_id) | 課程名(class_name) | 主修flag(main_class_flg) |
100 | 1 | 經濟學 | Y |
100 | 2 | 歷史學 | N |
200 | 2 | 歷史學 | N |
200 | 3 | 考古學 | Y |
200 | 4 | 計算機 | N |
300 | 4 | 計算機 | N |
400 | 5 | 化學 | N |
500 | 6 | 數學 | N |
現在我們要按照下面兩個條件對這個表進行查詢
1.只選修一門課程的人,返回那門課程的ID
2.選修多門課程的人,返回所選的主課程ID
簡單的想法就是,執行兩條不同的SQL語句進行查詢。
條件1
--條件1:只選擇了一門課程的學生
SELECT std_id, MAX(class_id) AS main_class FROM Studentclass GROUP BY std_id HAVINGCOUNT(*) = 1;
執行結果1 STD_ID MAIN_class 300 4 400 5 500 6
條件2 --條件2:選擇多門課程的學生 SELECT std_id, class_id AS main_class FROM Studentclass WHERE main_class_flg = 'Y' ;
執行結果2 STD_ID MAIN_class 100 1 200 3
如果使用Case函式,我們只要一條SQL語句就可以解決問題,具體如下所示 SELECT std_id, CASE WHEN COUNT(*) = 1 --只選擇一門課程的學生的情況 THEN MAX(class_id) ELSE MAX(CASE WHEN main_class_flg = 'Y' THEN class_id ELSE NULL END ) END AS main_class FROM Studentclass GROUP BY std_id;
執行結果 STD_ID MAIN_class 100 1 200 3 300 4 400 5 500 6 通過在Case函式中巢狀Case函式,在合計函式中使用Case函式等方法,我們可以輕鬆的解決這個問題。使用Case函式給我們帶來了更大的自由度。
最後提醒一下使用Case函式的新手注意不要犯下面的錯誤
CASE col_1 WHEN 1 THEN 'Right' WHEN NULL THEN 'Wrong' END
在這個語句中When Null這一行總是返回unknown,所以永遠不會出現Wrong的情況。因為這句實際表達的意思是
WHEN col_1 = NULL,這是一個錯誤的用法,這個時候我們應該選擇用WHEN col_1 IS NULL。
七、小結
select 與 case結合使用最大的好處有兩點,一是在顯示查詢結果時可以靈活的組織格式,二是有效避免了多次對同一個表或幾個表的訪問。
下面舉個簡單的例子來說明。例如表 students(id, name ,birthday, sex, grade),要求按每個年級統計男生和女生的數量各是多少,統計結果的表頭為,年級,男生數量,女生數量。如果不用select case when,為了將男女數量並列顯示,統計起來非常麻煩,先確定年級資訊,再根據年級取男生數和女生數,而且很容易出錯。
用select case when寫法如下:
SELECT grade, COUNT (CASE WHEN sex = 1 THEN 1
ELSE NULL
END) 男生數,
COUNT (CASE WHEN sex = 2 THEN
1
ELSE NULL
END) 女生數
FROM students
GROUP BY grade;
SUM函式,是SQL中用來累加的一個函式,SUM(1)這種帶數字引數的函式意思是,資料庫表中所有的列數*函式中的引數,
如:SELECT * FROM TEST 結果為10,
則SELECT SUM(1) FROM TEST 結果為10,SELECT SUM(2) FROM TEST 結果為20