ORACLE CASE WHEN THEN ELSE END語法


一、語法:case when then else end

Case具有兩種格式。簡單Case函數和Case搜索函數。

簡單Case函數

CASE sex
WHEN '1' THEN '男'
WHEN '2' THEN '女'
ELSE '其他' END

Case搜索函數 
CASE WHEN sex = '1' THEN '男' 
WHEN sex = '2' THEN '女' 
ELSE '其他' END  
 
說明:
1、簡單Case函數的寫法相對比較簡潔,但是和Case搜索函數相比,功能方面會有些限制,比如寫判斷式。
2、Case函數只返回第一個符合條件的值,剩下的Case部分將會被自動忽略。
 

二、Case函數一些實際場景

1、已知數據按照另外一種方式進行分組

----工資等級統計

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;

2、用一個SQL語句完成不同條件的分組。

--按照國家和性別進行分組,得出結果如下
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; 
CASE <wbr>WHEN <wbr>及 <wbr>SELECT <wbr>CASE <wbr>WHEN的用法
 
3、在Check中使用Case函數。 
  在Check中使用Case函數在很多情況下都是非常不錯的解決方法。可能有很多人根本就不用Check,那么我建議你在看過下面的例子之后也嘗試一下在SQL中使用Check。
  下面我們來舉個例子公司A,這個公司有個規定,女職員的工資必須高於1000塊。如果用Check和Case來表現的話,如下所示
THEN 1 ELSE 0 END  ELSE 1 END = 1 ) 
  如果單純使用Check: CONSTRAINT check_salary CHECK  ( sex = '2' AND salary > 1000 )  女職員的條件倒是符合了,男職員就無法輸入了。
4、根據條件有選擇的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 王五
假設有如上數據,需要把主鍵 ab相互交換。用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。需要注意的是,如果有需要交換主鍵的情況發生,多半是當初對這個表的設計進行得不夠到位,建議檢查表的設計是否妥當。
 
5、 兩個表數據是否一致的檢查。
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的情況。
 
6、在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
   有的學生選擇了同時修幾門課程(100,200)也有的學生只選擇了一門課程(300,400,500)。選修多門課程的學生,要選擇一門課程作為主修,主修flag里面寫入 Y。只選擇一門課程的學生,主修flag為N(實際上要是寫入Y的話,就沒有下面的麻煩事了,為了舉例子,還請多多包含)。
現在我們要按照下面兩個條件對這個表進行查詢

1.只選修一門課程的人,返回那門課程的ID

2.選修多門課程的人,返回所選的主課程ID

簡單的想法就是,執行兩條不同的SQL語句進行查詢。
條件1
--條件1:只選擇了一門課程的學生 

SELECT std_id, MAX(class_id) AS main_class  FROM Studentclass  GROUP BY std_id  HAVING COUNT(*) = 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。 

 

7、小結

   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;

 

 

***************************case when實際工作中的應用示例:

select a.order_id,
case a.prod_id
when 1 then
'普話'
when 140 then
'寬帶'
when 482 then
'vop'
end
from t_od_recv_queue_his a;


select to_char(dtrtime, 'yyyy-mm-dd HH24:mi') 時間,
count(case when a.ineid='221' then a.ineid else null end) as 網元221 ,
count(case when a.ineid='222' then a.ineid else null end) as 網元222 ,
count(case when a.ineid='223' then a.ineid else null end) as 網元223 ,
count(case when a.ineid='224' then a.ineid else null end) as 網元224 ,
count(case when a.ineid='225' then a.ineid else null end) as 網元225 ,
count(case when a.ineid='101' then a.ineid else null end) as 網元221 ,
count(case when a.ineid='102' then a.ineid else null end) as 網元222 ,
count(case when a.ineid='103' then a.ineid else null end) as 網元223 ,
count(case when a.ineid='104' then a.ineid else null end) as 網元224 ,
count(case when a.ineid='105' then a.ineid else null end) as 網元225
from tneticket_gsm a
where a.dtrtime > to_date('2014-03-06 00:00:00', 'yyyy-mm-dd HH24:mi:ss')
-- and a.dtrtime < to_date('2014-01-15 07:00:00', 'yyyy-mm-dd HH24:mi:ss')
and a.iworkstatus=1
and a.iworkresult=0
and a.ifinishflag=0
group by to_char(dtrtime, 'yyyy-mm-dd HH24:mi')
order by to_char(dtrtime, 'yyyy-mm-dd HH24:mi');

select to_char(t.dtrtime, 'yyyymm'),
sum(case when (b.dtrtime - t.dtrtime) * 24 * 3600< 1 then 1 else 0 end) 小於S,
sum(case when (b.dtrtime - t.dtrtime) * 24 * 3600>= 1 and (b.dtrtime - t.dtrtime) * 24 * 3600< 2 then 1 else 0 end) 小於1S,
sum(case when (b.dtrtime - t.dtrtime) * 24 * 3600>= 2 and (b.dtrtime - t.dtrtime) * 24 * 3600< 3 then 1 else 0 end) 小於S,
sum(case when (b.dtrtime - t.dtrtime) * 24 * 3600>= 3 and (b.dtrtime - t.dtrtime) * 24 * 3600< 4 then 1 else 0 end) 小於2S,
sum(case when (b.dtrtime - t.dtrtime) * 24 * 3600>= 4 and (b.dtrtime - t.dtrtime) * 24 * 3600< 5 then 1 else 0 end) 小於3S,
sum(case when (b.dtrtime - t.dtrtime) * 24 * 3600>= 5 and (b.dtrtime - t.dtrtime) * 24 * 3600< 6 then 1 else 0 end) 小於4S,
sum(case when (b.dtrtime - t.dtrtime) * 24 * 3600>= 6 and (b.dtrtime - t.dtrtime) * 24 * 3600< 8 then 1 else 0 end) 小於5S,
sum(case when (b.dtrtime - t.dtrtime) * 24 * 3600>= 8 and (b.dtrtime - t.dtrtime) * 24 * 3600< 10 then 1 else 0 end) 小於6S,
sum(case when (b.dtrtime - t.dtrtime) * 24 * 3600>= 10 then 1 else 0 end) 大於6S
from tworkrelation_gsm_near t,
(select a.vcwid, max(a.dtrtime) dtrtime
from tneticket_gsm_near a
where a.dtrtime > To_Date('2013-12-01', 'yyyy-mm-dd')
and a.vcsvccode in ('101240')--開戶
group by a.vcwid) b
where b.vcwid = t.vcwid
and t.dtrtime >
to_date('2013-10-01 00:00:00', 'yyyy-mm-dd hh24:mi:ss')
and t.dtrtime <
to_date('2013-10-31 23:59:59', 'yyyy-mm-dd hh24:mi:ss')
and (b.dtrtime - t.dtrtime) * 24 * 3600< 60--2min
and t.vcsla = 200
group by to_char(t.dtrtime, 'yyyymm')
order by to_char(t.dtrtime, 'yyyymm');

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM