《SQL必知必會》筆記3-過濾where滤否、and、or最仑、in藐俺、通配符

1 過濾檢索數(shù)據(jù)(WHERE)

1.1 使用WHERE子句

SELECT prod_name, prod_price 
FROM Products
WHERE prod_price = 3.49;

WHERE子句指定對搜索條件進(jìn)行過濾。

注意:在同時使用ORDER BY和WHERE子句時泥彤,應(yīng)該讓ORDER BY位于WHERE之后欲芹,否則會產(chǎn)生錯誤。


1.2 WHERE子句操作符

操作符 說明
= 等于
<> 不等于
!= 不等于
< 小于
<= 小于等于
!< 不小于
> 大于
>= 大于等于
!> 不大于
BETWEEN 在指定的兩個值之間
IS NULL 為NULL值

注意:表中列出的某些操作符是冗余的(如<>與!=相同吟吝,!<相當(dāng)于>=)菱父,并非所有的DBMS都支持這些操作符。


1.2.1 檢查單個值

列出所有價格小于10美元的產(chǎn)品剑逃。

SELECT prod_name, prod_price 
FROM Products
WHERE prod_price < 10;

1.2.2 不匹配檢查

列出所有不是供應(yīng)商DLL01制造的產(chǎn)品浙宜。

SELECT vend_id, prod_name 
FROM Products
WHERE vend_id <> 'DLL01';

SELECT vend_id, prod_name
FROM Products
WHERE vend_id != 'DLL01';

如果將值與字符串類型的列進(jìn)行比較,就需要限定引號蛹磺。用來與數(shù)值列進(jìn)行比較的值不用引號粟瞬。

!=和<>通常可以互換萤捆,但是并非所有DBMS都支持這兩種不等于操作符亩钟。


1.2.3 范圍值檢查

要檢查某個范圍的值,可以使用BETWEEN操作符鳖轰。例如:BETWEEN操作符可用來檢索價格在5美元和10美元之間的所有產(chǎn)品清酥,或在指定的開始日期和結(jié)束日期之間的所有日期。

檢索價格在5美元和10美元之間的所有產(chǎn)品蕴侣。

SELECT prod_name, prod_price 
FROM Products
WHERE prod_price BETWEEN 5 AND 10;

在使用BETWEEN時焰轻,必須指定兩個值——所需范圍的低端值和高端值。這兩個值必須用AND關(guān)鍵字分隔昆雀,BETWEEN匹配范圍中所有的值辱志,包括指定的開始值和結(jié)束值。


1.2.4 空值檢查

在創(chuàng)建表時狞膘,表設(shè)計人員可以指定其中的列能否不包含值揩懒。在一個列不包含值時,稱其包含空值NULL挽封。
NULL:無值(no value)已球,它與字段包含0、空字符串或僅僅包含空格不同。

SELECT cust_name 
FROM Customers 
WHERE cust_email IS NULL;

通過過濾選擇不包含指定值的所有行時智亮,你可能希望返回含NULL值的行忆某。但是做不到,因為未知(unknown)有特殊的含義阔蛉,數(shù)據(jù)庫不知道它們是否匹配弃舒,所以在進(jìn)行匹配過濾或非匹配過濾時,不會返回這些結(jié)果状原。

過濾數(shù)據(jù)時聋呢,一定要驗證被過濾列中含NULL的行確實出現(xiàn)在返回的數(shù)據(jù)中。


2 高級數(shù)據(jù)過濾(AND颠区、OR坝冕、IN、NOT IN)

2.1 組合WHERE子句

操作符:用來聯(lián)結(jié)或改變WHERE子句中的子句的關(guān)鍵字瓦呼,也稱為邏輯操作符(logical operator)喂窟。

2.1.1 AND操作符

要通過不止一個列進(jìn)行過濾,可以使用AND操作符給WHERE子句附加條件央串。

SELECT prod_id, prod_price, prod_name 
FROM Products
WHERE vend_id = 'DLL01' AND prod_price <= 4;

SELECT prod_id, prod_price, prod_name 
FROM Products
WHERE vend_id = 'DLL01' AND prod_price <= 4
ORDER BY prod_name;

這個例子只包含一個AND子句磨澡,因此最多有兩個過濾條件≈屎停可以增加多個過濾條件稳摄,每個條件間都要使用AND關(guān)鍵字。


2.1.2 OR操作符

許多DBMS在OR WHERE子句的第一個條件得到滿足的情況下饲宿,就不再計算第二個條件了(在第一個條件滿足時厦酬,不管第二個條件是否滿足,相應(yīng)的行都將被檢索出來)瘫想。

SELECT prod_name, prod_price 
FROM Products
WHERE vend_id = 'DLL01' OR vend_id = 'BRS01';

2.1.3 求值順序(AND仗阅、OR)

WHERE子句可以包含任意數(shù)目的AND和OR操作符,允許兩者結(jié)合以進(jìn)行復(fù)雜国夜、高級的過濾减噪。

假如需要列出價格為10美元及以上,且由DLL01或BRS01制造的所有產(chǎn)品车吹。

SELECT prod_name, prod_price 
FROM Products
WHERE vend_id = 'DLL01' OR vend_id = 'BRS01'
      AND prod_price >= 10;

相當(dāng)于:

SELECT prod_name, prod_price 
FROM Products
WHERE vend_id = 'DLL01' OR (vend_id = 'BRS01'
      AND prod_price >= 10);

返回的行中有4行價格小于10美元筹裕,顯然返回的行未按預(yù)期的進(jìn)行過濾。為什么會這樣呢窄驹?原因在于求值的順序朝卒。SQL在處理OR操作符前,優(yōu)先處理AND操作符乐埠。

當(dāng)SQL看到上述WHERE子句時抗斤,它理解為:由供應(yīng)商BRS01制造的價格為10美元以上的所有產(chǎn)品囚企,以及由供應(yīng)商DLL01制造的所有產(chǎn)品,而不管其價格如何豪治。換句話說,由于AND在求值過程中優(yōu)先級更高扯罐,操作符被錯誤地組合了负拟。

使用圓括號對操作符進(jìn)行明確分組。

SELECT prod_name, prod_price 
FROM Products
WHERE (vend_id = 'DLL01' OR vend_id = 'BRS01')
      AND prod_price >= 10;

這條SELECT語句與前一條的唯一差別是,將前兩個條件用圓括號括了起來。因為圓括號具有比AND或OR操作符更高的求值順序托嚣,所以DBMS首先過濾圓括號內(nèi)的OR條件罩息。這時,SQL語句變成了選擇由供應(yīng)商DLL01或BRS01制造的且價格在10美元及以上的所有產(chǎn)品蹋半,這正是我們希望的結(jié)果。


建議:任何時候使用具有AND和OR操作符的WHERE子句,都應(yīng)該使用圓括號明確地分組操作符谬墙。


2.2 IN操作符

IN操作符用來指定條件范圍,范圍中的每個條件都可以進(jìn)行匹配经备。IN取一組由逗號分隔拭抬、括在圓括號中的合法值。

檢索由供應(yīng)商DLL01和BRS01制造的所有產(chǎn)品侵蒙。

下面先使用IN操作符:

SELECT prod_name, prod_price 
FROM Products
WHERE vend_id IN ('DLL01', 'BRS01')
ORDER BY prod_name;

下面使用OR操作符:

SELECT prod_name, prod_price 
FROM Products
WHERE vend_id = 'DLL01' OR vend_id ='BRS01'
ORDER BY prod_name;

由上可以看出IN操作符完成了與OR相同的功能造虎。

為什么要使用IN操作符?其優(yōu)點如下:

  1. 在有很多合法選項時纷闺,IN操作符的語法更清楚算凿,更直觀。
  2. 在與其他AND和OR操作符組合使用IN時犁功,求值順序更容易管理氓轰。
  3. IN操作符一般比一組OR操作符執(zhí)行得更快(在上面這個合法選項很少的例子中,你看不出來性能差異)浸卦。
  4. IN的最大優(yōu)點是可以包含其他SELECT語句戒努,能夠更動態(tài)地建立WHERE子句。

2.3 NOT操作符

NOT操作符有且只有一個功能镐躲,那就是否定其后所跟的任何條件储玫。因為NOT從不單獨使用(它總是與其他操作符一起使用),所以它的語法與其他操作符有所不同萤皂。

NOT關(guān)鍵字可以用在要過濾的列前撒穷,而不僅是在其后。

列出除DLL01之外的所有供應(yīng)商制造的產(chǎn)品裆熙。

SELECT prod_name 
FROM Products
WHERE NOT vend_id = 'DLL01'
ORDER BY prod_name;

也可用<>操作符來完成端礼。

SELECT prod_name 
FROM Products
WHERE vend_id <> 'DLL01'
ORDER BY prod_name;

為什么使用NOT禽笑?

對于這里的這種簡單的WHERE子句,使用NOT確實沒有什么優(yōu)勢蛤奥,但在更復(fù)雜的子句中佳镜,NOT是非常有用的。例如凡桥,在與IN操作符聯(lián)合使用時蟀伸,NOT可以非常簡單地找出與條件列表不匹配的行。


3 用通配符進(jìn)行過濾(LIKE缅刽、%啊掏、_、[])

3.1 LIKE操作符

怎樣搜索產(chǎn)品名中包含文本bean bag的所有產(chǎn)品衰猛?用簡單的比較操作符肯定不行迟蜜,必須使用通配符。

利用通配符啡省,可以創(chuàng)建比較特定數(shù)據(jù)的搜索模式娜睛。在這個例子中,如果你想找出名稱包含bean bag的所有產(chǎn)品卦睹,可以構(gòu)造一個通配符搜索模式微姊,找出在產(chǎn)品名的任何位置出現(xiàn)bean bag的產(chǎn)品。

通配符(wildcard):用來匹配值的一部分的特殊字符分预。

搜索模式(search pattern):由字面值兢交、通配符或兩者組合構(gòu)成的搜索條件。

通配符本身實際上是SQL的WHERE子句中有特殊含義的字符笼痹,SQL支持幾種通配符配喳。為在搜索子句中使用通配符,必須使用LIKE操作符凳干。

通配符搜索只能用于文本字段(字符串)晴裹,非文本數(shù)據(jù)類型字段不能使用通配符搜索。

3.1.1 百分號(%)通配符

在搜索串中救赐,%表示任何字符出現(xiàn)任意次數(shù)涧团。

找到所有以詞Fish起頭的產(chǎn)品。

SELECT prod_id, prod_name 
FROM Products
WHERE prod_name LIKE 'Fish%';

通配符可在搜索模式中的任何位置使用经磅,并且可以使用多個通配符泌绣。

SELECT prod_id, prod_name 
FROM Products
WHERE prod_name LIKE '%bean bag%';

找出以F起頭、以y結(jié)尾的所有產(chǎn)品预厌。

SELECT prod_id, prod_name 
FROM Products
WHERE prod_name LIKE 'F%y';

簡單的解決辦法是給搜索模式再增加一個%號:'F%y'還匹配y之后的字符(或空格)阿迈,更好的辦法解決辦法是用函數(shù)去掉空格。

通配符%看起來像是可以匹配任何東西轧叽,但有個例外苗沧,這就是NULL刊棕。子句WHERE prod_name LIKE '%'不會匹配產(chǎn)品名稱為NULL的行。


3.1.2 下劃線(_)通配符

_只匹配單個字符待逞,而不是多個字符甥角。

SELECT prod_id, prod_name 
FROM Products
WHERE prod_name LIKE '__ inch teddy bear';

SELECT prod_id, prod_name 
FROM Products
WHERE prod_name LIKE '% inch teddy bear';

3.1.3 方括號([])通配符

方括號([])通配符用來指定一個字符集,它必須匹配指定位置(通配符的位置)的一個字符识樱。

此查詢只支持Access和SQL Server嗤无,不支持MySQL數(shù)據(jù)庫。

找出所有名字以J或M起頭的聯(lián)系人牺荠。

SELECT cust_contact 
FROM Customers
WHERE cust_contact LIKE '[JM]%'
ORDER BY cust_contact;

輸出結(jié)果為:

Jim Jones
John Smith
Michelle Green

3.2 使用通配符的技巧

  1. 不要過度使用通配符翁巍。如果其他操作符能達(dá)到相同的目的驴一,應(yīng)該使用其他操作符休雌。
  2. 在確實需要使用通配符時,盡量不要把它們用在搜索模式的開始處肝断。把通配符置于開始處杈曲,搜素起來是最慢的。
  3. 仔細(xì)注意通配符的位置胸懈。如果放錯位置担扑,可能不會返回想要的數(shù)據(jù)。

如果您發(fā)現(xiàn)文中有不清楚或者有問題的地方趣钱,請在下方評論區(qū)留言涌献,我會根據(jù)您的評論,更新文中相關(guān)內(nèi)容首有,謝謝燕垃!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市井联,隨后出現(xiàn)的幾起案子卜壕,更是在濱河造成了極大的恐慌,老刑警劉巖烙常,帶你破解...
    沈念sama閱讀 221,430評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件轴捎,死亡現(xiàn)場離奇詭異,居然都是意外死亡蚕脏,警方通過查閱死者的電腦和手機(jī)侦副,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,406評論 3 398
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來驼鞭,“玉大人跃洛,你說我怎么就攤上這事≈找椋” “怎么了汇竭?”我有些...
    開封第一講書人閱讀 167,834評論 0 360
  • 文/不壞的土叔 我叫張陵葱蝗,是天一觀的道長。 經(jīng)常有香客問我细燎,道長两曼,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,543評論 1 296
  • 正文 為了忘掉前任玻驻,我火速辦了婚禮悼凑,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘璧瞬。我一直安慰自己户辫,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 68,547評論 6 397
  • 文/花漫 我一把揭開白布嗤锉。 她就那樣靜靜地躺著渔欢,像睡著了一般。 火紅的嫁衣襯著肌膚如雪瘟忱。 梳的紋絲不亂的頭發(fā)上奥额,一...
    開封第一講書人閱讀 52,196評論 1 308
  • 那天,我揣著相機(jī)與錄音访诱,去河邊找鬼垫挨。 笑死,一個胖子當(dāng)著我的面吹牛触菜,可吹牛的內(nèi)容都是我干的九榔。 我是一名探鬼主播,決...
    沈念sama閱讀 40,776評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼涡相,長吁一口氣:“原來是場噩夢啊……” “哼哲泊!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起漾峡,我...
    開封第一講書人閱讀 39,671評論 0 276
  • 序言:老撾萬榮一對情侶失蹤攻旦,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后生逸,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體牢屋,經(jīng)...
    沈念sama閱讀 46,221評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,303評論 3 340
  • 正文 我和宋清朗相戀三年槽袄,在試婚紗的時候發(fā)現(xiàn)自己被綠了烙无。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,444評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡遍尺,死狀恐怖截酷,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情乾戏,我是刑警寧澤迂苛,帶...
    沈念sama閱讀 36,134評論 5 350
  • 正文 年R本政府宣布三热,位于F島的核電站,受9級特大地震影響三幻,放射性物質(zhì)發(fā)生泄漏就漾。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,810評論 3 333
  • 文/蒙蒙 一念搬、第九天 我趴在偏房一處隱蔽的房頂上張望抑堡。 院中可真熱鬧,春花似錦朗徊、人聲如沸首妖。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,285評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽有缆。三九已至,卻和暖如春舌仍,著一層夾襖步出監(jiān)牢的瞬間妒貌,已是汗流浹背通危。 一陣腳步聲響...
    開封第一講書人閱讀 33,399評論 1 272
  • 我被黑心中介騙來泰國打工铸豁, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人菊碟。 一個月前我還...
    沈念sama閱讀 48,837評論 3 376
  • 正文 我出身青樓节芥,卻偏偏與公主長得像,于是被迫代替她去往敵國和親逆害。 傳聞我的和親對象是個殘疾皇子头镊,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,455評論 2 359

推薦閱讀更多精彩內(nèi)容