sql server多字段字符串模糊查询存在字段null值查询失效问题

在 SQL Server 中进行多字段字符串模糊查询(如 WHERE col1 LIKE '%关键词%' OR col2 LIKE '%关键词%')时,如果某些字段为 NULL,会导致整个 OR 条件中该部分返回 UNKNOWN(非 TRUE/FALSE),从而漏掉本应匹配的记录,造成"查询失效"。

在 SQL 中,任何与 NULL 的比较(包括 LIKE)结果都是 UNKNOWN。

WHERE 子句只返回 TRUE 的行,UNKNOWN 和 FALSE 都会被过滤掉。

因此,如果 col1 IS NULL,那么 col1 LIKE '%xxx%' → UNKNOWN → 该行被排除,即使 col2 匹配!

sql 复制代码
-- 表数据:
-- id | name     | remark
-- 1  | 'Apple'  | NULL
-- 2  | NULL     | 'Banana'

-- 查询:找包含 'Apple' 的记录
SELECT * FROM t 
WHERE name LIKE '%Apple%' OR remark LIKE '%Apple%';
-- 结果:✅ 返回 id=1

-- 查询:找包含 'Banana' 的记录
SELECT * FROM t 
WHERE name LIKE '%Banana%' OR remark LIKE '%Banana%';
-- 结果:❌ **返回空!** 因为 name IS NULL → name LIKE ... = UNKNOWN,remark 虽匹配但 OR 整体未生效?
-- 实际上:remark LIKE '%Banana%' 是 TRUE,所以应该返回 id=2 ✅

-- ❗ 真正的问题场景是:
-- 当你搜索的关键词在非 NULL 字段中不存在,而另一个字段是 NULL 时,可能误判。
-- 但更常见的是:开发者误以为 NULL 会"跳过",其实不会影响其他 OR 条件。

上述 OR 查询 本身是正确的!只要有一个条件为 TRUE,整行就会被返回。

真正的问题通常出现在以下场景:

场景 1:使用 AND 而不是 OR

sql 复制代码
-- 错误:要求两个字段都匹配(但其中一个为 NULL)
WHERE name LIKE '%xxx%' AND remark LIKE '%xxx%'
-- 如果 remark IS NULL,则 AND 整体为 UNKNOWN → 行被过滤

场景 2:动态拼接 SQL 时未处理 NULL

sql 复制代码
// Java 伪代码
String sql = "SELECT * FROM t WHERE 1=1";
if (keyword != null) {
    sql += " AND (name LIKE '%" + keyword + "%' OR remark LIKE '%" + keyword + "%')";
}

这本身没问题,但如果字段有 NULL,不影响 OR 逻辑。

场景 3:误用 NOT LIKE + NULL

sql 复制代码
-- 想排除包含 'test' 的记录
WHERE name NOT LIKE '%test%'
-- 如果 name IS NULL,条件为 UNKNOWN → 行被排除!
-- 但你可能希望保留 NULL 值的行

正确解决方案:安全处理 NULL

虽然 OR 查询本身对 NULL 是安全的,但为了代码健壮性和避免逻辑误解,建议显式处理 NULL。

方法 1:使用 ISNULL() 或 COALESCE() 替换 NULL

sql 复制代码
-- 将 NULL 视为空字符串 '',确保 LIKE 可执行
SELECT * FROM t
WHERE ISNULL(name, '') LIKE '%关键词%'
   OR ISNULL(remark, '') LIKE '%关键词%';

优点:逻辑清晰;

避免任何 UNKNOWN;

兼容所有 SQL Server 版本。

方法 2:显式检查 NULL(更标准)

sql 复制代码
SELECT * FROM t
WHERE (name LIKE '%关键词%' OR name IS NULL)
   OR (remark LIKE '%关键词%' OR remark IS NULL);
-- ❌ 这是错误的!因为 OR name IS NULL 会让所有 NULL 行都返回!

-- ✅ 正确写法(仅当字段非 NULL 时才匹配):
SELECT * FROM t
WHERE (name IS NOT NULL AND name LIKE '%关键词%')
   OR (remark IS NOT NULL AND remark LIKE '%关键词%');

但注意:这和 ISNULL(col, '') LIKE ...效果相同,只是写法不同。

推荐写法(生产环境)

sql 复制代码
-- 方案 A:简洁版(推荐)
WHERE ISNULL(col1, '') LIKE '%关键词%'
   OR ISNULL(col2, '') LIKE '%关键词%'
   OR ISNULL(col3, '') LIKE '%关键词%';

-- 方案 B:显式版(可读性高)
WHERE (col1 LIKE '%关键词%' OR col1 IS NULL) -- ❌ 不要这样写!
-- 正确显式版:
WHERE (col1 IS NOT NULL AND col1 LIKE '%关键词%')
   OR (col2 IS NOT NULL AND col2 LIKE '%关键词%');

强烈推荐方案 A:ISNULL(col, '') LIKE ...代码简短;

性能几乎无损(除非字段极大);

行为符合直觉(NULL 视为"不包含关键词")。

性能提醒

LIKE '%关键词%'无法使用索引,大数据量时很慢;

ISNULL(col, '') 也不会破坏性能(因为本来就不能用索引);

如需高性能全文搜索,考虑 SQL Server Full-Text Search:

sql 复制代码
WHERE CONTAINS((col1, col2), '关键词')

动态 SQL(MyBatis / 应用层)最佳实践

sql 复制代码
<select id="search" resultType="Entity">
  SELECT * FROM t
  WHERE 1=1
  <if test="keyword != null and keyword != ''">
    AND (
      ISNULL(name, '') LIKE CONCAT('%', #{keyword}, '%')
      OR ISNULL(remark, '') LIKE CONCAT('%', #{keyword}, '%')
    )
  </if>
</select>

注意:SQL Server 用 + 拼接字符串,但 MyBatis 推荐用 CONCAT(兼容性好)或参数化。

记住:

NULL LIKE ... 永远不为 TRUE,用 ISNULL 或 COALESCE 将其转为可比较的值是最安全的做法。

相关推荐
无籽西瓜a2 小时前
详解Redis持久化:RDB、AOF与混合持久化
数据库·redis·缓存
人工智能知识库2 小时前
华为HCCDA-GaussDB题库(带详细解析)
数据库·华为·gaussdb·题库·hccda-gaussdb·hccda
齐 飞2 小时前
数据库批量插入耗时过长问题rewriteBatchedStatements=true
数据库·mysql
sg_knight2 小时前
SQL 中的 IFNULL 函数是什么?
数据库·sql·mysql·oracle·database·关系型数据库·db
程序员黄老师2 小时前
一分钟了解时序数据库(TSDB)
大数据·数据库·时序数据库
你才是臭弟弟2 小时前
时序数据库TDengine TSDB(安装/介绍)
数据库·时序数据库·tdengine
正在走向自律2 小时前
从“可用”到“好用”:金仓时序数据库开发者体验深度测评与二次开发生态搭建指南
数据库·时序数据库·金仓数据库
1.14(java)2 小时前
事务操作全解析:ACID特性与实战技巧
数据库·事务·acid特性
熊文豪2 小时前
国产化替代浪潮下:金仓时序数据库的破局之路
数据库·时序数据库·金仓数据库