揭開SQL中NULL的神秘面紗
這篇文章將揭開 SQL 中 NULL 的神秘面紗。這個問題可能困擾著很多初級開發(fā)者。
在查詢數(shù)據(jù)庫時,如果你想知道一個列是否為 NULL,SQL 查詢語句該怎么寫呢?
是不是這樣:
SELECT * FROM SOME_TABLE WHERE SOME_COLUMN = NULL
或者這樣寫:
SELECT * FROM SOME_TABLE WHERE SOME_COLUMN IS NULL
正確的寫法應該是第二種(WHERE SOME_COLUMN IS NULL)。
為什么要這樣寫?
在進行數(shù)據(jù)庫數(shù)據(jù)比較操作時,我們不會使用“IS”關(guān)鍵詞,不是嗎?
例如,如果我們想要知道一個列的值是否等于 1,WHERE 語句是這樣的:
WHERE SOME_COLUMN = 1
那為什么 NULL 值要用 IS 關(guān)鍵字呢?為什么要以這種方式來處理 NULL?
因為,在 SQL 中,NULL 表示“未知”。也就是說,NULL 值表示的是“未知”的值。
1NULL 等于未知
在大多數(shù)數(shù)據(jù)庫中,NULl 和空字符串是有區(qū)別的。
但并不是所有數(shù)據(jù)庫都這樣,例如,Oracle 就不支持空字符串,它會把空字符串自動轉(zhuǎn)成 NULL 值。
在其他大多數(shù)數(shù)據(jù)庫里,NULL 值和字符串的處理方式是不一樣的:
- 空字符串表示“沒有值”,這個值是已知的。
- NULL 表示“未知值”,這個值是未知的。
這就好比我問了一個問題:“美國總統(tǒng)羅斯福的中間名是什么”?
- 可能有人會回答說:“我不知道羅斯福總統(tǒng)的中間名是什么”。對于這種情況,可以在數(shù)據(jù)庫中使用 MIDDLE_NAME 列來表示羅斯福的中間名,而這一列的值為 NULL。
- 也可能有人會回答說:“羅斯??偨y(tǒng)沒有中間名。他的父母沒有給他取中間名,所以我知道羅斯??偨y(tǒng)確實沒有中間名”。對于這種情況,MIDDLE_NAME 列應該是一個空字符串。
Oracle 比較特殊,兩個值都使用 NULL 來表示,而其他大多數(shù)數(shù)據(jù)庫會區(qū)分對待。
但只要記住 NULL 表示的是一個未知的值,那么在寫 SQL 查詢語句時就會得心應手。
例如,如果你有一個這樣的查詢語句:
SELECT * FROM SOME_TABLE WHERE 1 = 1
這個查詢會返回所有的行(假設 SOME_TABLE 不是空表),因為表達式“1=1”一定為 true。
如果我這樣寫:
SELECT * FROM SOME_TABLE WHERE 1 = 0
表達式“1=0”是 false,這個查詢語句不會返回任何數(shù)據(jù)。
但如果我寫成這樣:
SELECT * FROM SOME_TABLE WHERE 1 = NULL
這個時候,數(shù)據(jù)庫不知道這兩個值(1 和 NULL)是否相等,所以它也不會返回任何數(shù)據(jù)。
2三元邏輯
SQL 查詢語句中的 WHERE 一般會有三種結(jié)果:
- 它可以是 true(這個時候會返回數(shù)據(jù));
- 它可以是 false(這個時候不會返回數(shù)據(jù));
- 它也可以是 NULL 或未知(這個時候也不會返回數(shù)據(jù))。
你可能會想:“既然這樣,那我為什么要去關(guān)心是 false 還是 NULL?它們不是都不會返回數(shù)據(jù)嗎?”
接下來,我來告訴你在哪些情況下會有問題:我們來看看 NOT()。
假設有這樣的一個查詢語句:
SELECT * FROM SOME_TABLE WHERE NOT(1 = 1)
數(shù)據(jù)庫首先會計算 1=1,這個顯然是 true。
接著,數(shù)據(jù)庫會應用 NOT() 條件,所以 WHERE 返回 false。
所以,上面的查詢不會返回任何數(shù)據(jù)。
但如果把語句改成這樣:
SELECT * FROM SOME_TABLE WHERE NOT(1 = 0)
數(shù)據(jù)庫首先會計算 1=0,這個肯定是 false。
接著,數(shù)據(jù)庫應用 NOT() 條件,這樣就得到相反的結(jié)果,變成了 true。
所以,這個語句會返回數(shù)據(jù)。
但如果把語句再改成下面這樣呢?
SELECT * FROM SOME_TABLE WHERE NOT(1 = NULL)
數(shù)據(jù)庫首先計算 1=NULL,它不知道 1 是否等于 NULL,因為它不知道 NULL 的值是什么。
所以,這個計算不會返回 true,也不會返回 false,它會返回一個 NULL。
接下來,NOT() 會繼續(xù)解析上一個計算返回的結(jié)果。
當 NOT() 遇到 NULL,它會生成另一個 NULL。未知的相反面是另一個未知。
所以,對于這兩個查詢:
SELECT * FROM SOME_TABLE WHERE NOT(1 = NULL)
和
SELECT * FROM SOME_TABLE WHERE 1 = NULL
都不會返回數(shù)據(jù),盡管它們是完全相反的。
3NULL 和 NOT IN
如果我有這樣的一個查詢語句:
SELECT * FROM SOME_TABLE WHERE 1 IN (1, 2, 3, 4, NULL)
很顯然,WHERE 返回 true,這個語句將返回數(shù)據(jù),因為 1 在括號列表里是存在的。
但如果這么寫:
SELECT * FROM SOME_TABLE WHERE 1 NOT IN (1, 2, 3, 4, NULL)
很顯然,WHERE 返回 false,這個查詢不會返回數(shù)據(jù),因為 1 在括號列表里存在,但我們說的是“NOT IN”。
但如果我們把語句改成這樣呢?
SELECT * FROM SOME_TABLE WHERE 5 NOT IN (1, 2, 3, 4, NULL)
這里的 WHRE 不會返回數(shù)據(jù),因為它的結(jié)果不是 true。數(shù)字 5 在括號列表里可能不存在,也可能存在,因為當中有一個 NULL 值(數(shù)據(jù)庫不知道 NULL 的值是什么)。
這個 WHERE 會返回 NULL,所以整個查詢不會返回任何數(shù)據(jù)。
希望你們現(xiàn)在都清楚該怎么在 SQL 語句中處理 NULL 值了。
到此這篇關(guān)于揭開SQL中NULL的神秘面紗的文章就介紹到這了,更多相關(guān)SQL NULL內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MySQL降權(quán)運行之MySQL以Guests帳戶啟動設置方法
我們在windows服務器中使用mysql數(shù)據(jù)的時候,mysql默認都是以system權(quán)限運行,如果出現(xiàn)了安全問題,黑客就可以通過mysql提權(quán)新建用戶什么的,所以mysql低權(quán)限運行非常必要2014-07-07
mysql8.0.11 winx64安裝配置方法圖文教程(win10)
這篇文章主要為大家詳細介紹了win10下mysql8.0.11 winx64安裝配置方法圖文教程,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-05-05
Linux下MySQL5.7.18二進制包安裝教程(無默認配置文件my_default.cnf)
這篇文章主要介紹了Linux下MySQL5.7.18二進制包安裝教程(無默認配置文件my_default.cnf) ,需要的朋友可以參考下2017-05-05
MySQL中create table as 與like的區(qū)別分析
這篇文章主要介紹了MySQL中create table as 與like的區(qū)別,結(jié)合實例分析了二者在使用中的具體區(qū)別與主要用途,需要的朋友可以參考下2016-01-01
MySQL數(shù)據(jù)庫備份和還原的常用命令小結(jié)
MySQL數(shù)據(jù)庫備份和還原的常用命令小結(jié),學習mysql的朋友可以參考下2012-03-03

