TSQL--NULL值和三值逻辑

在SQL SERVER 中逻辑表达式存在三种值:TRUE+FALSE+UNKNOWN。UNKNOW可以理解为不确定,既不是TRUE又不是FALSE的表达式,主要由与NULL相关的逻辑判断引起,值为NULL就意味着该未赋值或该值未确定。

与NULL值做算术运算时,其结果是NULL,如果1+NULL结果为NULL

 

SQL Server不同场景下对UNKNOWN处理不同,对NULL的处理也不同。

1. 在WHERE+ON+HAVING三种筛选器中,所有运算结果非TURE(FALSE 与UNKNOW)的记录都不会返回;

2. 在CHECK约束中,所有运算结果为非FALSE(TRUE与UNKNOW)的都属于满足CHECK约束的;

3. 在UNIQUE约束中,如果列定义未限制为NOT NULL,那么允许该列存在一条NULL值,如果另外插入或更新一条记录为NULL时,会违法UNIQUE约束,NULL与NULL是相等的;

4. 在GROUP BY中,NULL值被认为相同而分为一组,NULL与NULL是相等的;

5. 在ORDER BY中,NULL值被认为相同而排列在一起,所有NULL值比已知值小,NULL与NULL是相等的;

 

除上述3/4/5条中提到的情况外,NULL与NULL是不相等的。

默认情况下,即SET ANSI_NULLS ON时,对于条件(WHERE C1 = NULL)这种查询,C1列值为NULL的行不会被返回;

而当SET ANSI_NULLS ON时,对于条件(WHERE C1 = NULL)这种查询,C1列值为NULL的行会被返回,此时C1=NULL应该被理解为做C1 IS NULL 运算,而不应该理解为NULL与NULL相等;

理解混乱的同学可以做以下测试:

 

--==========================================
--生成测试数据
DECLARE @TB TABLE
(
    C1 INT
)

INSERT INTO @TB
SELECT 1
UNION
SELECT NULL

--===========================================
--修改默认选项值,设置ANSI_NULLS
--注意该设置是回话级别,而不是批处理级别或语句级别
SET ANSI_NULLS OFF;

--============================================
--当ANSI_NULLS OFF时,C1=NULL 等同于C1 IS NULL
--因此查询返回一条NULL的记录
SELECT * FROM @TB
WHERE C1=NULL

--============================================
--当ANSI_NULLS OFF,NULL与NULL仍不认为相等
--因此查询中不会返回NULL的记录
SELECT * FROM @TB AS TB1
INNER JOIN @TB TB2
ON TB1.C1=TB2.C1

运行结果为:

为规范操作和避免混乱,强烈建议使用IS NULL 和IS NOT NULL判断值是否为NULL,避免修改默认选项ANSI_NULLS为OFF。

 

除上面提到的特殊情况外,由于NULL与NULL是不相等的,因此

1. 在做IN和EXISTS运算如WHERE C1 IN(SELECT ID FROM TB1) 或者 WHERE EXISTS (SELECT ID FROM TB1 WHERE ID =C1)时,所有C1列为NULL的行都不会被返回,无论表TB1的ID列是否存在NULL值;

2. 对NOT IN运算如T1.C1 NOT IN (SELECT T2.C1 FROM T2) 时,如果表T2的C1列存在NULL值, 那么查询将不会返回任何记录,无论表T1的C1列是否有NULL值存在。

3. 对NOT EXISTS运算如NOT EXISTS(SELECT C1 FROM T2 WHERE T2.C1=T1.C1),会返回T1表有但T2表没有且T1.C2 IS NOT NULL的记录

测试DEMO

--=================
--生成测试数据
DECLARE @TB TABLE
(
    C1 INT
)

DECLARE @TB2 TABLE
(
    C1 INT
)


INSERT INTO @TB
SELECT 1
UNION
SELECT 2


INSERT INTO @TB2
SELECT 1
UNION
SELECT NULL

--=================================
--NOT EXISTS返回记录为2的行
SELECT * FROM @TB AS TB1
WHERE NOT EXISTS(
    SELECT C1 FROM @TB2 AS TB2 
    WHERE TB2.C1=TB1.C1)
--================================
--NOT IN 不返回任何行
SELECT * FROM @TB
WHERE C1 NOT IN
(
    SELECT C1 FROM @TB2
)
View Code

 

因此,IN和EXIST可以相互改写,但是NOT IN不能随便改写为NOT EXISTS.

 

PS: 对于运算结果为UNKNOW的逻辑表达式,再做NOT运算,结果仍未UNKNOW

 --==========================================================

妹子振贴

posted on 2014-01-14 18:08  笑东风  阅读(788)  评论(0编辑  收藏  举报

导航