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

相关推荐
晚风_END7 小时前
Linux|操作系统|最新版openzfs编译记录
linux·运维·服务器·数据库·spring·中间件·个人开发
dLYG DUMS7 小时前
DBeaver连接本地MySQL、创建数据库表的基础操作
数据库·mysql
苍煜8 小时前
MySQL分库分表和ES到底怎么选?
数据库·mysql·elasticsearch
茉莉玫瑰花茶8 小时前
Qt 信号与槽 [ 1 ]
开发语言·数据库·qt
czlczl200209259 小时前
松散索引扫描/跳跃索引扫描
数据库·mysql·性能优化
星马梦缘10 小时前
数据库作战记录 实验7、8
数据库·sql·oracle
安逸sgr11 小时前
Hermes Agent + Obsidian 打造第二大脑(六):分层记忆系统的设计逻辑——L0/L1/L2/L3 四层记忆详解
数据库·agent·知识库·hermes·hermesagent
苍煜11 小时前
一篇讲懂分库分表:概念、spirngboot实战
数据库·oracle
梦想画家11 小时前
PostgreSQL 物化视图实战:从数据固化到智能刷新的全链路指南
数据库·postgresql·物化视图
weoptions11 小时前
简单sql注入中如何通过简单语句判断注入类型&注入方法
数据库·sql