sqlserver中Case的使用方法(上下篇)第2/2頁(yè)
四,根據(jù)條件有選擇的UPDATE。
例,有如下更新條件
工資5000以上的職員,工資減少10%
工資在2000到4600之間的職員,工資增加15%
很容易考慮的是選擇執(zhí)行兩次UPDATE語(yǔ)句,如下所示
--條件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;
但是事情沒(méi)有想象得那么簡(jiǎn)單,假設(shè)有個(gè)人工資5000塊。首先,按照條件1,工資減少10%,變成工資4500。接下來(lái)運(yùn)行第二個(gè)SQL時(shí)候,因?yàn)檫@個(gè)人的工資是4500在2000到4600的范圍之內(nèi),需增加15%,最后這個(gè)人的工資結(jié)果是5175,不但沒(méi)有減少,反而增加了。如果要是反過(guò)來(lái)執(zhí)行,那么工資4600的人相反會(huì)變成減少工資。暫且不管這個(gè)規(guī)章是多么荒誕,如果想要一個(gè)SQL 語(yǔ)句實(shí)現(xiàn)這個(gè)功能的話,我們需要用到Case函數(shù)。代碼如下:
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;
這里要注意一點(diǎn),最后一行的ELSE salary是必需的,要是沒(méi)有這行,不符合這兩個(gè)條件的人的工資將會(huì)被寫(xiě)成NUll,那可就大事不妙了。在Case函數(shù)中Else部分的默認(rèn)值是NULL,這點(diǎn)是需要注意的地方。
這種方法還可以在很多地方使用,比如說(shuō)變更主鍵這種累活。
一般情況下,要想把兩條數(shù)據(jù)的Primary key,a和b交換,需要經(jīng)過(guò)臨時(shí)存儲(chǔ),拷貝,讀回?cái)?shù)據(jù)的三個(gè)過(guò)程,要是使用Case函數(shù)的話,一切都變得簡(jiǎn)單多了。
p_key | col_1 | col_2 |
a | 1 | 張三 |
b | 2 | 李四 |
c | 3 | 王五 |
假設(shè)有如上數(shù)據(jù),需要把主鍵a
和b
相互交換。用Case函數(shù)來(lái)實(shí)現(xiàn)的話,代碼如下
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');
同樣的也可以交換兩個(gè)Unique key。需要注意的是,如果有需要交換主鍵的情況發(fā)生,多半是當(dāng)初對(duì)這個(gè)表的設(shè)計(jì)進(jìn)行得不夠到位,建議檢查表的設(shè)計(jì)是否妥當(dāng)。
五,兩個(gè)表數(shù)據(jù)是否一致的檢查。
Case函數(shù)不同于DECODE函數(shù)。在Case函數(shù)中,可以使用BETWEEN,LIKE,IS NULL,IN,EXISTS等等。比如說(shuō)使用IN,EXISTS,可以進(jìn)行子查詢,從而 實(shí)現(xiàn)更多的功能。
下面具個(gè)例子來(lái)說(shuō)明,有兩個(gè)表,tbl_A,tbl_B,兩個(gè)表中都有keyCol列?,F(xiàn)在我們對(duì)兩個(gè)表進(jìn)行比較,tbl_A中的keyCol列的數(shù)據(jù)如果在tbl_B的keyCol列的數(shù)據(jù)中可以找到, 返回結(jié)果'Matched',如果沒(méi)有找到,返回結(jié)果'Unmatched'。
要實(shí)現(xiàn)下面這個(gè)功能,可以使用下面兩條語(yǔ)句
--使用IN的時(shí)候
SELECT keyCol,
CASE WHEN keyCol IN ( SELECT keyCol FROM tbl_B )
THEN 'Matched'
ELSE 'Unmatched' END Label
FROM tbl_A;
--使用EXISTS的時(shí)候
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的結(jié)果是相同的。也可以使用NOT IN和NOT EXISTS,但是這個(gè)時(shí)候要注意NULL的情況。
六,在Case函數(shù)中使用合計(jì)函數(shù)
假設(shè)有下面一個(gè)表
學(xué)號(hào)(std_id) | 課程ID(class_id) | 課程名(class_name) | 主修flag(main_class_flg) |
100 | 1 | 經(jīng)濟(jì)學(xué) | Y |
100 | 2 | 歷史學(xué) | N |
200 | 2 | 歷史學(xué) | N |
200 | 3 | 考古學(xué) | Y |
200 | 4 | 計(jì)算機(jī) | N |
300 | 4 | 計(jì)算機(jī) | N |
400 | 5 | 化學(xué) | N |
500 | 6 | 數(shù)學(xué) | N |
有的學(xué)生選擇了同時(shí)修幾門(mén)課程(100,200)也有的學(xué)生只選擇了一門(mén)課程(300,400,500)。選修多門(mén)課程的學(xué)生,要選擇一門(mén)課程作為主修,主修flag里面寫(xiě)入Y。只選擇一門(mén)課程的學(xué)生,主修flag為N(實(shí)際上要是寫(xiě)入Y的話,就沒(méi)有下面的麻煩事了,為了舉例子,還請(qǐng)多多包含)。
現(xiàn)在我們要按照下面兩個(gè)條件對(duì)這個(gè)表進(jìn)行查詢
- 只選修一門(mén)課程的人,返回那門(mén)課程的ID
- 選修多門(mén)課程的人,返回所選的主課程ID
簡(jiǎn)單的想法就是,執(zhí)行兩條不同的SQL語(yǔ)句進(jìn)行查詢。
條件1
--條件1:只選擇了一門(mén)課程的學(xué)生
SELECT std_id, MAX(class_id) AS main_class
FROM Studentclass
GROUP BY std_id
HAVING COUNT(*) = 1;
執(zhí)行結(jié)果1
STD_ID MAIN_class
------ ----------
300 4
400 5
500 6
條件2
--條件2:選擇多門(mén)課程的學(xué)生
SELECT std_id, class_id AS main_class
FROM Studentclass
WHERE main_class_flg = 'Y' ;
執(zhí)行結(jié)果2
STD_ID MAIN_class
------ ----------
100 1
200 3
如果使用Case函數(shù),我們只要一條SQL語(yǔ)句就可以解決問(wèn)題,具體如下所示
SELECT std_id,
CASE WHEN COUNT(*) = 1 --只選擇一門(mén)課程的學(xué)生的情況
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;
運(yùn)行結(jié)果
STD_ID MAIN_class
------ ----------
100 1
200 3
300 4
400 5
500 6
通過(guò)在Case函數(shù)中嵌套Case函數(shù),在合計(jì)函數(shù)中使用Case函數(shù)等方法,我們可以輕松的解決這個(gè)問(wèn)題。使用Case函數(shù)給我們帶來(lái)了更大的自由度。
最后提醒一下使用Case函數(shù)的新手注意不要犯下面的錯(cuò)誤
CASE col_1
WHEN 1 THEN 'Right'
WHEN NULL THEN 'Wrong'
END
在這個(gè)語(yǔ)句中When Null這一行總是返回unknown,所以永遠(yuǎn)不會(huì)出現(xiàn)Wrong的情況。因?yàn)檫@句可以替換成WHEN col_1 = NULL,這是一個(gè)錯(cuò)誤的用法,這個(gè)時(shí)候我們應(yīng)該選擇用WHEN col_1 IS NULL。
相關(guān)文章
SQL Server復(fù)制刪除發(fā)布時(shí)遇到錯(cuò)誤18752的問(wèn)題及解決方法
朋友反饋他無(wú)法刪除一臺(tái)SQL Server數(shù)據(jù)庫(kù)上的發(fā)布,具體情況為刪除一個(gè)SQL Server Replication的發(fā)布時(shí),遇到這樣的錯(cuò)誤問(wèn)題如何解決呢,下面小編給大家分享SQL Server復(fù)制刪除發(fā)布時(shí)遇到錯(cuò)誤18752的問(wèn)題及解決方法,感興趣的朋友一起看看吧2024-01-01SQL Server中參數(shù)化SQL寫(xiě)法遇到parameter sniff ,導(dǎo)致不合理執(zhí)行計(jì)劃重用的快速解決方法
這篇文章主要介紹了SQL Server中參數(shù)化SQL寫(xiě)法遇到parameter sniff ,導(dǎo)致不合理執(zhí)行計(jì)劃重用的快速解決方法的相關(guān)資料,需要的朋友可以參考下2016-07-07SQL SERVER使用ODBC 驅(qū)動(dòng)建立的鏈接服務(wù)器調(diào)用存儲(chǔ)過(guò)程時(shí)參數(shù)不能為NULL值
這篇文章主要介紹了SQL SERVER使用ODBC 驅(qū)動(dòng)建立的鏈接服務(wù)器調(diào)用存儲(chǔ)過(guò)程時(shí)參數(shù)不能為NULL值的相關(guān)資料,需要的朋友可以參考下2016-01-01SQLserver 數(shù)據(jù)庫(kù)危險(xiǎn)存儲(chǔ)過(guò)程刪除與恢復(fù)方法
SQLserver 數(shù)據(jù)庫(kù)危險(xiǎn)存儲(chǔ)過(guò)程刪除與恢復(fù)方法,有時(shí)候是為了安全,也為了以后使用方便2011-09-09sqlserver 數(shù)據(jù)庫(kù)連接字符串中的可選項(xiàng)收集
sqlserver 數(shù)據(jù)庫(kù)連接字符串中的可選項(xiàng)收集,需要的朋友可以參考下。2011-10-10SQL中的單條件判斷函數(shù)IF和多條件判斷CASE WHEN的用法
MySQL提供了IF、IFNULL、CASE等條件判斷函數(shù),本文就來(lái)介紹一下SQL中的單條件判斷函數(shù)IF和多條件判斷CASE WHEN的用法,感興趣的可以了解一下2023-10-10一個(gè)完整的SQL SERVER數(shù)據(jù)庫(kù)全文索引的示例介紹
以下是介紹SQL SERVER數(shù)據(jù)庫(kù)全文索引的示例,以pubs數(shù)據(jù)庫(kù)為例。需要的朋友參考下2013-07-07SQL Server存儲(chǔ)過(guò)程生成insert語(yǔ)句實(shí)例
這篇文章主要介紹了SQL Server使用存儲(chǔ)過(guò)程生成insert語(yǔ)句再執(zhí)行大量插入數(shù)據(jù)的方法2013-11-11SQL?Server數(shù)據(jù)庫(kù)中已存在名為'student'對(duì)象的解決辦法
這篇文章主要給大家介紹了關(guān)于SQL?Server數(shù)據(jù)庫(kù)中已存在名為'student'對(duì)象的解決辦法,解決方法很簡(jiǎn)單,并且也很實(shí)用,不止有這一個(gè)用處,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下2023-11-11