【问题处理】—— SQL里进行 null值 的判断永远返回 false?

简介: 【问题处理】—— SQL里进行 null值 的判断永远返回 false?

项目场景:

一个SQL查询的数据条目少了,本来应该有6条数据,但现在只有5条


问题描述

一个SQL查询的数据条目少了,本来应该有6条数据,但现在只有五条

SELECT
    intent.business_date,
    intent.intent_id,
    detail.approve_status
  FROM
    intention intent
    LEFT JOIN approvedetail detail ON intent.intent_id = detail.intent_id 
 where 
    detail.approve_status != '4' ;

本身是个非常简单的左连接查询,但是只查出来5条

而主表有6条数据

SELECT
    intent.business_date,
    intent.intent_id
  FROM
    intention intent;


原因分析:

因为我们只用了一个where条件,所以问题肯定在这里,那是不是approvedetail 里有等于4的数据,所以被过滤掉了呢?我们来看这张表的数据

select intent_id, approve_status from oapprovedetail ;


1d679212a0a446caa3ac3a22feed6bfe.png

可以看到有五条数据,但里面并没有状态为‘4’的行。那么从逻辑上说,左表不应该被过滤,应该是6条数据啊,也就是说预想中的查询数据应该如下,第6行数据approve_status应该为null,而不是整行被过滤掉


此刻,心里隐隐有了推测。会不会是null值的判断有问题?其实这个问题很早有听闻过,但是时间太长没什么印象了。说的就是在sql 里对 null 进行值的判断得到的都是false,即使你的判断是 (null != ‘4’)这种不等于判断

那么我们接下来直接开始印证这个结论

SELECT 
  CASE WHEN
    ( NULL != '4' ) 
  THEN
      1 ELSE 2 
  END AS 'res' ;

d3a414e3cc7148f3ac4910943a4859f9.png

果然出来的结果是2 ,也就是说 ( NULL != ‘4’ ) 在数据库里的判断是 false, 上述结论是在mysql 里是这样的,oracle 呢?我们也可以试一下


SELECT 
  CASE WHEN
    ( NULL != '4' ) 
  THEN
      1 ELSE 2 
  END AS res from dual;

oracle 里的结果也是2

后面,我又调换等于号两边,对 ( ‘4’ != null)进行了判断,结果还是2。那么看来这个逻辑主要就是涉及 null 值的判断 都会返回 false 了


解决方案:

知道了问题的原因,如果想查出第6行数据,那就只能针对这种特殊情况做补丁了,最后的结果是

SELECT
    intent.business_date,
    intent.intent_id,
    detail.approve_status
  FROM
    intention intent
    LEFT JOIN approvedetail detail ON intent.intent_id = detail.intent_id 
 where 
    detail.approve_status != '4' or detail.approve_status is null;

最后结果符合预期

目录
打赏
0
0
0
0
241
分享
相关文章
SQL NOT NULL
【11月更文挑战第14天】
79 6
|
7月前
|
SQL
SQL NOT NULL 约束
【7月更文挑战第18天】SQL NOT NULL 约束。
65 6
|
7月前
|
SQL
SQL NOT NULL 约束
【7月更文挑战第16天】SQL NOT NULL 约束。
45 3
|
8月前
|
SQL
SQL NULL 值
SQL NULL 值
397 3
SQL NULL 函数
SQL NULL 函数
48 1
MySQL设计规约问题之为什么应该把字段定义为NOT NULL并且提供默认值
MySQL设计规约问题之为什么应该把字段定义为NOT NULL并且提供默认值
|
8月前
|
SQL
SQL NOT NULL 约束
SQL NOT NULL 约束
54 1
实时计算 Flink版产品使用合集之从MySQL同步数据到Doris时,历史数据时间字段显示为null,而增量数据部分的时间类型字段正常显示的原因是什么
实时计算Flink版作为一种强大的流处理和批处理统一的计算框架,广泛应用于各种需要实时数据处理和分析的场景。实时计算Flink版通常结合SQL接口、DataStreamAPI、以及与上下游数据源和存储系统的丰富连接器,提供了一套全面的解决方案,以应对各种实时计算需求。其低延迟、高吞吐、容错性强的特点,使其成为众多企业和组织实时数据处理首选的技术平台。以下是实时计算Flink版的一些典型使用合集。
python在mysql中插入或者更新null空值
这段代码是Python操作MySQL数据库的示例。它执行SQL查询从表`a_kuakao_school`中选取`id`,`university_id`和`grade`,当`university_id`大于0时按升序排列。然后遍历结果,根据`row[4]`的值决定`grade`是否为`NULL`。若不为空,`grade`被格式化为字符串;否则,设为`NULL`。接着构造UPDATE语句更新`university`表中对应`id`的`grade`值,并提交事务。重要的是,字符串`NULL`不应加引号,否则更新会失败。
201 2
在 MySQL 中使用 IS NULL
【8月更文挑战第12天】
725 0
在 MySQL 中使用 IS NULL

热门文章

最新文章