SQL Server 2022 新语法:IS [NOT] DISTINCT FROM 彻底解决 NULL 比较难题

在日常开发中,NULL 值的比较一直是 SQL Server 的痛点之一。NULL = NULL 返回的不是 TRUE 而是 NULL,导致很多场景下需要写繁琐的 IS NULL 判断。SQL Server 2022 引入了 IS [NOT] DISTINCT FROM 语法,彻底解决了这个问题。

老写法的痛点

先看一个典型场景:比较两列值是否相等,其中可能包含 NULL。

测试数据:

复制代码
CREATE TABLE #TestData (
    ID      INT,
    ColA    NVARCHAR(50),
    ColB    NVARCHAR(50)
)

INSERT INTO #TestData VALUES
(1, 'Apple',  'Apple'),
(2, 'Apple',  'Banana'),
(3, NULL,     NULL),
(4, NULL,     'Apple'),
(5, 'Apple',  NULL)

如果不处理NULL:

复制代码
SELECT *
FROM   #TestData
WHERE  ColA = ColB

结果,字段为NULL的没有搜索出来:

老写法(需要手动处理 NULL):

复制代码
-- 判断两列"相等"(含 NULL 相等的语义)
SELECT *
FROM   #TestData
WHERE  (ColA = ColB) OR (ColA IS NULL AND ColB IS NULL)

逻辑没问题,但写法很繁琐。当列增多时,代码可读性急剧下降。


SQL Server 2022 新写法:IS NOT DISTINCT FROM

复制代码
-- 新写法,语义完全一致,简洁清晰
SELECT *
FROM   #TestData
WHERE  ColA IS NOT DISTINCT FROM ColB

核心语义:

  • A IS NOT DISTINCT FROM B:A 和 B 相等(NULL = NULL 视为相等),等价于 (A = B) OR (A IS NULL AND B IS NULL)
  • A IS DISTINCT FROM B:A 和 B 不相等,等价于 NOT ((A = B) OR (A IS NULL AND B IS NULL))

反向用法:IS DISTINCT FROM

复制代码
-- 找出两列值不同的行(含 NULL 不同的语义)
SELECT *
FROM   #TestData
WHERE  ColA IS DISTINCT FROM ColB

在 UPDATE 中的典型应用

这个语法在数据同步场景中非常有用。只更新真正有变化的行(包括 NULL 变动):

复制代码
CREATE TABLE #Source (ID INT, Val NVARCHAR(50))
CREATE TABLE #Target (ID INT, Val NVARCHAR(50))

INSERT INTO #Source VALUES (1, 'A'), (2, NULL), (3, 'C')
INSERT INTO #Target VALUES (1, 'A'), (2, 'B'),  (3, NULL)

-- 只更新值发生变化的行
UPDATE T
SET    T.Val = S.Val
FROM   #Target T
JOIN   #Source S ON T.ID = S.ID
WHERE  T.Val IS DISTINCT FROM S.Val

与 NULL 直接比较

IS [NOT] DISTINCT FROM 也支持与字面量(包括 NULL)比较:

复制代码
SELECT *
FROM   #TestData
WHERE  ColA IS NOT DISTINCT FROM NULL   -- 等价于 ColA IS NULL

SELECT *
FROM   #TestData
WHERE  ColA IS DISTINCT FROM NULL       -- 等价于 ColA IS NOT NULL

总结

IS [NOT] DISTINCT FROM 是 SQL Server 2022 中一个非常实用的语法补充,解决了长期以来 NULL 值比较需要写冗长条件的问题。特别是在以下场景中推荐使用:

  1. 数据同步/对比时,判断两列是否真正发生了变化
  2. 参数化查询中,参数值可能为 NULL 时的等值比较
  3. 需要将 NULL = NULL 视为相等的业务场景

代码简洁,语义明确,值得在 SQL Server 2022 环境中推广使用。

相关推荐
~ rainbow~2 小时前
前端转型全栈(四)——常见的错误及解决方案
数据库·oracle·全栈
数厘2 小时前
2.1SQL 学习:先懂数据库概念再学 SQL
数据库·sql·学习
Cat_Rocky3 小时前
redis哨兵模式
数据库·redis
广师大-Wzx3 小时前
一篇文章看懂MySQL数据库(下)
java·开发语言·数据结构·数据库·windows·python·mysql
hef2884 小时前
golang如何使用range over func_golang range over func迭代器使用方法
jvm·数据库·python
qq_380619165 小时前
html如何查看windows
jvm·数据库·python
爱学习的小邓同学5 小时前
MySQL --- MySQL数据库基础
数据库·mysql
wgzrmlrm745 小时前
如何加固SQL环境部署_删除默认安装的示例数据库
jvm·数据库·python
雨墨✘6 小时前
golang如何实现设备指纹识别_golang设备指纹识别实现详解
jvm·数据库·python