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 环境中推广使用。

相关推荐
jiayou6435 分钟前
KingbaseES 表级与列级加密完全指南
数据库·后端
GBASE20 小时前
G术时刻 |GBase 8s数据库事务并发控制之封锁技术介绍(下)
数据库
xiezhr1 天前
逛GitHub发现了一款免费的带AI功能的数据库管理工具
数据库·ai编程·dba
吃糖的小孩2 天前
给 QQ AI 机器人设计“可控记忆”:会话摘要、手动长期记忆与角色卡边界
数据库
笃行3503 天前
金仓数据库数据安全双防线:静态存储加密与传输加密实战
数据库
笃行3503 天前
金仓数据库物理备份实战:sys_rman 全流程演练与误覆盖抢救
数据库
笃行3503 天前
金仓数据库逻辑备份实战:从全库导出到 Schema 替换的完整闭环
数据库
SelectDB4 天前
阶跃星辰基于 SelectDB 构建 PB 级 Agent 可观测平台
大数据·数据库·aigc
这个DBA有点耶4 天前
GROUP BY优化全解:如何写出既不丢数据又飞快的分组查询
数据库·mysql·架构