在 SQL 語句中處理 NULL 值的方法
在日常使用數(shù)據(jù)庫時(shí),你在意過NULL值么?
其實(shí),NULL值在數(shù)據(jù)庫中是一個很特殊且有趣的存在,下面我們一起來看看吧;
在查詢數(shù)據(jù)庫時(shí),如果你想知道一個列(例如:用戶注冊年限 USER_AGE)是否為 NULL,SQL 查詢語句該怎么寫呢?
是這樣:
SELECT * FROM TABLE WHERE USER_AGE = NULL
還是這樣?
SELECT * FROM TABLE WHERE USER_AGE IS NULL
當(dāng)然,正確的寫法應(yīng)該是第二種(WHERE USER_AGE IS NULL)。
但為什么要這樣寫呢?在進(jìn)行數(shù)據(jù)庫數(shù)據(jù)比較操作時(shí),我們不會使用“IS”關(guān)鍵詞,不是嗎?

例如,如果我們想要知道一個列的值是否等于 1,WHERE 語句是這樣的:
WHERE USER_AGE = 1
那為什么 NULL 值要用 IS 關(guān)鍵字呢?為什么要以這種方式來處理 NULL?
因?yàn)?,?SQL 中,NULL 表示“未知”。也就是說,NULL 值表示的是“未知”的值。
NULL = 未知;
在大多數(shù)數(shù)據(jù)庫中,NULl 和空字符串是有區(qū)別的。
但并不是所有數(shù)據(jù)庫都這樣,例如,Oracle 就不支持空字符串,它會把空字符串自動轉(zhuǎn)成 NULL 值。
在其他大多數(shù)數(shù)據(jù)庫里,NULL 值和字符串的處理方式是不一樣的:
- 空字符("")串雖然表示“沒有值”,但這個值是已知的。
- NULL 表示 “未知值”,這個值是未知的。
Oracle 比較特殊,兩個值都使用 NULL 來表示,而其他大多數(shù)數(shù)據(jù)庫會區(qū)分對待。
但只要記住 NULL 表示的是一個未知的值,那么在寫 SQL 查詢語句時(shí)就會得心應(yīng)手。
例如,如果你有一個這樣的查詢語句:
SELECT * FROM SOME_TABLE WHERE 1 = 1
這個查詢會返回所有的行(假設(shè) SOME_TABLE 不是空表),因?yàn)楸磉_(dá)式“1=1”一定為 true。
如果我這樣寫:
SELECT * FROM SOME_TABLE WHERE 1 = 0
表達(dá)式“1=0”是 false,這個查詢語句不會返回任何數(shù)據(jù)。
但如果我寫成這樣:
SELECT * FROM SOME_TABLE WHERE 1 = NULL
這個時(shí)候,數(shù)據(jù)庫不知道這兩個值(1 和 NULL)是否相等,因此會認(rèn)定為“NULL”或“未知”,所以它也不會返回任何數(shù)據(jù)。
三元邏輯
SQL 查詢語句中的 WHERE 一般會有三種結(jié)果:
- 它可以是 true(這個時(shí)候會返回?cái)?shù)據(jù));
- 它可以是 false(這個時(shí)候不會返回?cái)?shù)據(jù));
- 它也可以是 NULL 或未知(這個時(shí)候也不會返回?cái)?shù)據(jù));

你可能會想:“既然這樣,那我為什么要去關(guān)心是 false 還是 NULL?它們不是都不會返回?cái)?shù)據(jù)嗎?”
接下來,我來告訴你在哪些情況下會有問題:我們來看看 NOT( ) 方法。
假設(shè)有這樣的一個查詢語句:
SELECT * FROM SOME_TABLE WHERE NOT(1 = 1)
數(shù)據(jù)庫首先會計(jì)算 1=1,這個顯然是 true。
接著,數(shù)據(jù)庫會應(yīng)用 NOT() 條件,所以 WHERE 返回 false。
所以,上面的查詢不會返回任何數(shù)據(jù)。
但如果把語句改成這樣:
SELECT * FROM SOME_TABLE WHERE NOT(1 = 0)
數(shù)據(jù)庫首先會計(jì)算 1=0,這個肯定是 false。
接著,數(shù)據(jù)庫應(yīng)用 NOT() 條件,這樣就得到相反的結(jié)果,變成了 true。
所以,這個語句會返回?cái)?shù)據(jù)。
但如果把語句再改成下面這樣呢?
SELECT * FROM SOME_TABLE WHERE NOT(1 = NULL)
數(shù)據(jù)庫首先計(jì)算 1=NULL,它不知道 1 是否等于 NULL,因?yàn)樗恢?NULL 的值是什么。
所以,這個計(jì)算不會返回 true,也不會返回 false,它會返回一個 NULL。
接下來,NOT() 會繼續(xù)解析上一個計(jì)算返回的結(jié)果。
當(dāng) NOT() 遇到 NULL,它會生成另一個 NULL。未知的相反面是另一個未知。
所以,對于這兩個查詢:
SELECT * FROM SOME_TABLE WHERE NOT(1 = NULL) SELECT * FROM SOME_TABLE WHERE 1 = NULL
都不會返回?cái)?shù)據(jù),盡管它們是完全相反的。
NULL 和 NOT IN
如果我有這樣的一個查詢語句:
SELECT * FROM TABLE WHERE 1 IN (1, 2, 3, 4, NULL)
很顯然,WHERE 返回 true,這個語句將返回?cái)?shù)據(jù),因?yàn)?1 在括號列表里是存在的。
但如果這么寫:
SELECT * FROM SOME_TABLE WHERE 1 NOT IN (1, 2, 3, 4, NULL)
很顯然,WHERE 返回 false,這個查詢不會返回?cái)?shù)據(jù),因?yàn)?1 在括號列表里存在,但我們說的是“NOT IN”。
但如果我們把語句改成這樣呢?
SELECT * FROM SOME_TABLE WHERE 5 NOT IN (1, 2, 3, 4, NULL)
這里的 WHERE 不會返回?cái)?shù)據(jù),因?yàn)樗慕Y(jié)果不是 true。數(shù)字 5 在括號列表里可能不存在,也可能存在,因?yàn)楫?dāng)中有一個 NULL 值(數(shù)據(jù)庫不知道 NULL 的值是什么)。
這個 WHERE 會返回 NULL,所以整個查詢不會返回任何數(shù)據(jù)。
希望大家現(xiàn)在都清楚該怎么在 SQL 語句中處理 NULL 值了。
以上就是在 SQL 語句中處理 NULL 值的方法的詳細(xì)內(nèi)容,更多關(guān)于SQL 中的 NULL值的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
SQL?SERVER自動備份以及自動清除設(shè)置圖文教程
作為一名DBA、運(yùn)維人員、實(shí)施人員亦或是測試人員必須要學(xué)會如何備份以及清理備份文件,自動清理備份文件,這篇文章主要給大家介紹了關(guān)于SQL?SERVER自動備份以及自動清除設(shè)置的相關(guān)資料,文中通過圖文介紹的非常詳細(xì),需要的朋友可以參考下2024-02-02
SQLServer查詢某個時(shí)間段購買過商品的所有用戶
這篇文章主要介紹了SQLServer查詢某個時(shí)間段購買過商品的所有用戶,需要的朋友可以參考下2017-07-07
SQLSERVER全文目錄全文索引的使用方法和區(qū)別講解
這篇文章主要介紹了SQLSERVER全文目錄全文索引的使用方法,有圖有代碼,大家參考使用吧2013-11-11
SQL Report Builder 報(bào)表里面的常見問題分析
這篇文章主要介紹了SQL Report Builder 報(bào)表里面的常見問題分析的相關(guān)資料,需要的朋友可以參考下2015-12-12
sql server多行數(shù)據(jù)拼接的實(shí)例方法
sql server多行數(shù)據(jù)拼接的實(shí)例方法,需要的朋友可以參考一下2013-04-04
Sql Server 2012 分頁方法分析(offset and fetch)
最近在分析 Sql Server 2012 中 offset and fetch 的新特性,發(fā)現(xiàn) offset and fetch 無論語法的簡潔還是功能的強(qiáng)大,都是相當(dāng)相當(dāng)不錯的2012-08-08

