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

相关推荐
Micro麦可乐2 小时前
Spring Boot 实战:从零设计一个短链系统(含完整代码与数据库设计)
数据库·spring boot·后端·哈希算法·雪花算法·短链系统
码农阿豪2 小时前
从零到一:Spring Boot快速接入金仓数据库实战
数据库·spring boot·后端
鼎讯信通2 小时前
风电光缆运维提质增效:G-4000A 光缆故障追踪仪破解风场巡检难题
运维·网络·数据库
三十..3 小时前
MySQL 从入门到高可用架构实战精要
运维·数据库·mysql
cfm_29143 小时前
Redis五大基本数据结构底层了解
数据结构·数据库·redis
真实的菜4 小时前
Redis 从入门到精通(十二):典型业务场景实战 —— 排行榜、限流器、秒杀系统、Session 共享
数据库·redis·python
你想考研啊4 小时前
mysql数据库导出导入
数据库·mysql·oracle
十年编程老舅5 小时前
Linux DRM:底层逻辑与实践架构
数据库·mysql
The Sheep 20235 小时前
Vue复习
linux·服务器·数据库
云边有个稻草人5 小时前
深度解析:KingbaseES高可用架构落地原理与生产运维实战
数据库·读写分离·数据库运维·金仓数据库·国产数据库技术·数据备份恢复