當前位置: 妍妍網 > 碼農

在 SQL 中該如何處理 NULL 值,你真的清楚麽?

2024-05-14碼農

點選「 IT碼徒 」, 關註,置頂 公眾號

每日技術幹貨,第一時間送達!

在日常使用資料庫時,你在意過NULL值麽?

其實,NULL值在資料庫中是一個很特殊且有趣的存在,下面我們一起來看看吧;

1

前言

在查詢資料庫時,如果你想知道一個列(例如:使用者註冊年限 USER_AGE)是否為 NULL,SQL 查詢語句該怎麽寫呢?

是這樣:

SELECT * FROMTABLEWHERE USER_AGE = NULL

還是這樣?

SELECT * FROMTABLEWHERE USER_AGE ISNULL

當然,正確的寫法應該是第二種( WHERE USER_AGE IS NULL )。

但為什麽要這樣寫呢?在進行資料庫數據比較操作時,我們不會使用「IS」關鍵詞,不是嗎?

例如,如果我們想要知道一個列的值是否等於 1,WHERE 語句是這樣的:

WHERE USER_AGE = 1

那為什麽 NULL 值要用 IS 關鍵字呢?為什麽要以這種方式來處理 NULL?

因為, 在 SQL 中,NULL 表示「未知」。也就是說,NULL 值表示的是「未知」的值。

NULL = 未知;

大多數資料庫中,NULL 和空字串是有區別的

但並不是所有資料庫都這樣,例如,Oracle 就不支持空字串,它會把空字串自動轉成 NULL 值。

在其他大多數資料庫裏,NULL 值和字串的處理方式是不一樣的:

  • 空字元("")串雖然表示「沒有值」,但這個值是已知的。

  • NULL 表示 「未知值」,這個值是未知的。

  • 這就好比我問了一個問題:「川建國的小名叫什麽?」

    有人會回答說:「我不知道川建國的小名是什麽」。對於這種情況,可以在資料庫中使用 Nickname 列來表示川建國的小名,而這一列的值為 NULL。

    也有人會回答說:「川建國沒有小名。他的父母沒有給他取小名,大家雖然一直叫他川二狗,但是我知道川建國確實沒有小名」。對於這種情況, Nickname 列應該是一個空字串("")。

    Oracle 比較特殊,兩個值都使用 NULL 來表示,而其他大多數資料庫會區分對待。

    但只要記住 NULL 表示的是一個未知的值,那麽在寫 SQL 查詢語句時就會得心應手。

    例如,如果你有一個這樣的查詢語句:

    SELECT * FROM SOME_TABLE WHERE1 = 1

    這個查詢會返回所有的行(假設 SOME_TABLE 不是空表),因為運算式「1=1」一定為 true。

    如果我這樣寫:

    SELECT * FROM SOME_TABLE WHERE1 = 0

    運算式「1=0」是 false,這個查詢語句不會返回任何數據。

    但如果我寫成這樣:

    SELECT * FROM SOME_TABLE WHERE1 = NULL

    這個時候,資料庫不知道這兩個值(1 和 NULL)是否相等,因此會認定為「NULL」或「未知」,所以它也不會返回任何數據。

    2

    三元邏輯

    SQL 查詢語句中的 WHERE 一般會有三種結果:

  • 它可以是 true(這個時候會返回數據);

  • 它可以是 false(這個時候不會返回數據);

  • 它也可以是 NULL 或未知(這個時候也不會返回數據);

  • 你可能會想:「既然這樣,那我為什麽要去關心是 false 還是 NULL?它們不是都不會返回數據嗎?」

    接下來,我來告訴你在哪些情況下會有問題:我們來看看 NOT( ) 方法。

    假設有這樣的一個查詢語句:

    SELECT * FROM SOME_TABLE WHERENOT(1 = 1)

    資料庫首先會計算 1=1,這個顯然是 true。

    接著,資料庫會套用 NOT() 條件,所以 WHERE 返回 false。

    所以,上面的查詢不會返回任何數據。

    但如果把語句改成這樣:

    SELECT * FROM SOME_TABLE WHERENOT(1 = 0)

    資料庫首先會計算 1=0,這個肯定是 false。

    接著,資料庫套用 NOT() 條件,這樣就得到相反的結果,變成了 true。

    所以,這個語句會返回數據。

    但如果把語句再改成下面這樣呢?

    SELECT * FROM SOME_TABLE WHERENOT(1 = NULL)

    資料庫首先計算 1=NULL,它不知道 1 是否等於 NULL,因為它不知道 NULL 的值是什麽。

    所以,這個計算不會返回 true,也不會返回 false,它會返回一個 NULL。

    接下來,NOT() 會繼續解析上一個計算返回的結果。

    當 NOT() 遇到 NULL,它會生成另一個 NULL。未知的相反面是另一個未知。

    所以,對於這兩個查詢:

    SELECT * FROM SOME_TABLE WHERENOT(1 = NULL)
    SELECT * FROM SOME_TABLE WHERE1 = NULL

    都不會返回數據,盡管它們是完全相反的。

    3

    NULL 和 NOT IN

    如果我有這樣的一個查詢語句:

    SELECT * FROMTABLEWHERE1IN (1234NULL)

    很顯然,WHERE 返回 true,這個語句將返回數據,因為 1 在括弧列表裏是存在的。

    但如果這麽寫:

    SELECT * FROM SOME_TABLE WHERE1NOTIN (1234NULL)

    很顯然,WHERE 返回 false,這個查詢不會返回數據,因為 1 在括弧列表裏存在,但我們說的是「NOT IN」。

    但如果我們把語句改成這樣呢?

    SELECT * FROM SOME_TABLE WHERE5NOTIN (1234NULL)

    這裏的 WHERE 不會返回數據,因為它的結果不是 true。數位 5 在括弧列表裏可能不存在,也可能存在,因為當中有一個 NULL 值(資料庫不知道 NULL 的值是什麽)。

    這個 WHERE 會返回 NULL,所以整個查詢不會返回任何數據。

    希望大家現在都清楚該怎麽在 SQL 語句中處理 NULL 值了。

    END

    PS:防止找不到本篇文章,可以收藏點贊,方便翻閱尋找哦。

    往期推薦