1 過濾檢索數據(WHERE)
1.1 使用WHERE子句
SELECT prod_name, prod_price
FROM Products
WHERE prod_price = 3.49;
WHERE子句指定對搜索條件進行過濾。
注意:在同時使用ORDER BY和WHERE子句時,應該讓ORDER BY位于WHERE之后,否則會產生錯誤。
1.2 WHERE子句操作符
操作符 | 說明 |
---|---|
= | 等于 |
<> | 不等于 |
!= | 不等于 |
< | 小于 |
<= | 小于等于 |
!< | 不小于 |
> | 大于 |
>= | 大于等于 |
!> | 不大于 |
BETWEEN | 在指定的兩個值之間 |
IS NULL | 為NULL值 |
注意:表中列出的某些操作符是冗余的(如<>與!=相同,!<相當于>=),并非所有的DBMS都支持這些操作符。
1.2.1 檢查單個值
列出所有價格小于10美元的產品。
SELECT prod_name, prod_price
FROM Products
WHERE prod_price < 10;
1.2.2 不匹配檢查
列出所有不是供應商DLL01制造的產品。
SELECT vend_id, prod_name
FROM Products
WHERE vend_id <> 'DLL01';
SELECT vend_id, prod_name
FROM Products
WHERE vend_id != 'DLL01';
如果將值與字符串類型的列進行比較,就需要限定引號。用來與數值列進行比較的值不用引號。
!=和<>通??梢曰Q,但是并非所有DBMS都支持這兩種不等于操作符。
1.2.3 范圍值檢查
要檢查某個范圍的值,可以使用BETWEEN操作符。例如:BETWEEN操作符可用來檢索價格在5美元和10美元之間的所有產品,或在指定的開始日期和結束日期之間的所有日期。
檢索價格在5美元和10美元之間的所有產品。
SELECT prod_name, prod_price
FROM Products
WHERE prod_price BETWEEN 5 AND 10;
在使用BETWEEN時,必須指定兩個值——所需范圍的低端值和高端值。這兩個值必須用AND關鍵字分隔,BETWEEN匹配范圍中所有的值,包括指定的開始值和結束值。
1.2.4 空值檢查
在創建表時,表設計人員可以指定其中的列能否不包含值。在一個列不包含值時,稱其包含空值NULL。
NULL:無值(no value),它與字段包含0、空字符串或僅僅包含空格不同。
SELECT cust_name
FROM Customers
WHERE cust_email IS NULL;
通過過濾選擇不包含指定值的所有行時,你可能希望返回含NULL值的行。但是做不到,因為未知(unknown)有特殊的含義,數據庫不知道它們是否匹配,所以在進行匹配過濾或非匹配過濾時,不會返回這些結果。
過濾數據時,一定要驗證被過濾列中含NULL的行確實出現在返回的數據中。
2 高級數據過濾(AND、OR、IN、NOT IN)
2.1 組合WHERE子句
操作符:用來聯結或改變WHERE子句中的子句的關鍵字,也稱為邏輯操作符(logical operator)。
2.1.1 AND操作符
要通過不止一個列進行過濾,可以使用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關鍵字。
2.1.2 OR操作符
許多DBMS在OR WHERE子句的第一個條件得到滿足的情況下,就不再計算第二個條件了(在第一個條件滿足時,不管第二個條件是否滿足,相應的行都將被檢索出來)。
SELECT prod_name, prod_price
FROM Products
WHERE vend_id = 'DLL01' OR vend_id = 'BRS01';
2.1.3 求值順序(AND、OR)
WHERE子句可以包含任意數目的AND和OR操作符,允許兩者結合以進行復雜、高級的過濾。
假如需要列出價格為10美元及以上,且由DLL01或BRS01制造的所有產品。
SELECT prod_name, prod_price
FROM Products
WHERE vend_id = 'DLL01' OR vend_id = 'BRS01'
AND prod_price >= 10;
相當于:
SELECT prod_name, prod_price
FROM Products
WHERE vend_id = 'DLL01' OR (vend_id = 'BRS01'
AND prod_price >= 10);
返回的行中有4行價格小于10美元,顯然返回的行未按預期的進行過濾。為什么會這樣呢?原因在于求值的順序。SQL在處理OR操作符前,優先處理AND操作符。
當SQL看到上述WHERE子句時,它理解為:由供應商BRS01制造的價格為10美元以上的所有產品,以及由供應商DLL01制造的所有產品,而不管其價格如何。換句話說,由于AND在求值過程中優先級更高,操作符被錯誤地組合了。
使用圓括號對操作符進行明確分組。
SELECT prod_name, prod_price
FROM Products
WHERE (vend_id = 'DLL01' OR vend_id = 'BRS01')
AND prod_price >= 10;
這條SELECT語句與前一條的唯一差別是,將前兩個條件用圓括號括了起來。因為圓括號具有比AND或OR操作符更高的求值順序,所以DBMS首先過濾圓括號內的OR條件。這時,SQL語句變成了選擇由供應商DLL01或BRS01制造的且價格在10美元及以上的所有產品,這正是我們希望的結果。
建議:任何時候使用具有AND和OR操作符的WHERE子句,都應該使用圓括號明確地分組操作符。
2.2 IN操作符
IN操作符用來指定條件范圍,范圍中的每個條件都可以進行匹配。IN取一組由逗號分隔、括在圓括號中的合法值。
檢索由供應商DLL01和BRS01制造的所有產品。
下面先使用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操作符?其優點如下:
- 在有很多合法選項時,IN操作符的語法更清楚,更直觀。
- 在與其他AND和OR操作符組合使用IN時,求值順序更容易管理。
- IN操作符一般比一組OR操作符執行得更快(在上面這個合法選項很少的例子中,你看不出來性能差異)。
- IN的最大優點是可以包含其他SELECT語句,能夠更動態地建立WHERE子句。
2.3 NOT操作符
NOT操作符有且只有一個功能,那就是否定其后所跟的任何條件。因為NOT從不單獨使用(它總是與其他操作符一起使用),所以它的語法與其他操作符有所不同。
NOT關鍵字可以用在要過濾的列前,而不僅是在其后。
列出除DLL01之外的所有供應商制造的產品。
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確實沒有什么優勢,但在更復雜的子句中,NOT是非常有用的。例如,在與IN操作符聯合使用時,NOT可以非常簡單地找出與條件列表不匹配的行。
3 用通配符進行過濾(LIKE、%、_、[])
3.1 LIKE操作符
怎樣搜索產品名中包含文本bean bag的所有產品?用簡單的比較操作符肯定不行,必須使用通配符。
利用通配符,可以創建比較特定數據的搜索模式。在這個例子中,如果你想找出名稱包含bean bag的所有產品,可以構造一個通配符搜索模式,找出在產品名的任何位置出現bean bag的產品。
通配符(wildcard):用來匹配值的一部分的特殊字符。
搜索模式(search pattern):由字面值、通配符或兩者組合構成的搜索條件。
通配符本身實際上是SQL的WHERE子句中有特殊含義的字符,SQL支持幾種通配符。為在搜索子句中使用通配符,必須使用LIKE操作符。
通配符搜索只能用于文本字段(字符串),非文本數據類型字段不能使用通配符搜索。
3.1.1 百分號(%)通配符
在搜索串中,%表示任何字符出現任意次數。
找到所有以詞Fish起頭的產品。
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結尾的所有產品。
SELECT prod_id, prod_name
FROM Products
WHERE prod_name LIKE 'F%y';
簡單的解決辦法是給搜索模式再增加一個%號:'F%y'還匹配y之后的字符(或空格),更好的辦法解決辦法是用函數去掉空格。
通配符%看起來像是可以匹配任何東西,但有個例外,這就是NULL。子句WHERE prod_name LIKE '%'不會匹配產品名稱為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數據庫。
找出所有名字以J或M起頭的聯系人。
SELECT cust_contact
FROM Customers
WHERE cust_contact LIKE '[JM]%'
ORDER BY cust_contact;
輸出結果為:
Jim Jones
John Smith
Michelle Green
3.2 使用通配符的技巧
- 不要過度使用通配符。如果其他操作符能達到相同的目的,應該使用其他操作符。
- 在確實需要使用通配符時,盡量不要把它們用在搜索模式的開始處。把通配符置于開始處,搜素起來是最慢的。
- 仔細注意通配符的位置。如果放錯位置,可能不會返回想要的數據。
如果您發現文中有不清楚或者有問題的地方,請在下方評論區留言,我會根據您的評論,更新文中相關內容,謝謝!