解读SQL中的Null 超详细教程

Null表示的是一种未知状态,未来状态,比如小明兜里有多少钱我不清楚,但也不能肯定为0,这时在计算机中就使用Null来表示未知和不确定。

  虽然熟练掌握SQL的人对于Null不会有什么疑问,但总结得很全的文章还是很难找,看到一篇英文版的, 感觉还不错。

  Tony Hoare 在1965年发明了 null 引用, 并认为这是他犯下的“几十亿美元的错误”. 即便是50年后的今天, SQL中的 null 值还是导致许多常见错误的罪魁祸首.

  我们一起来看那些最令人震惊的情况。

  Null不支持大小/相等判断

下面的2个查询,不管表 users 中有多少条记录,返回的记录都是0行:

select * from users where deleted_at = null;
? result: 0 rows
select * from users where deleted_at != null;
? result: 0 rows

  怎么会这样子? 一切只因为 null 是表示一种“未知”的类型。也就是说,用常规的比较操作符(normal conditional operators)来将 null 与其他值比较是没有意义的。 Null 也不等于 Null(近似理解: 未知的值不能等于未知的值,两者间的关系也是未知,否则数学和逻辑上就乱套了)。

  ? 注意: 下面的SQL适合于MySQL,如果是Oracle,你需要加上 … from dual;

select null > 0;
? result: null
select null < 0;
? result: null
select null = 0;
? result: null
select null = null;
? result: null
select null != null;
? result: null

  将某个值与 null 进行比较的正确方法是使用 is 关键字, 以及 is not 操作符:

select * from users
where deleted_at is null;
? result: 所有被标记为删除的 users
select * from users
where deleted_at is not null;

? result: 所有被标记为删除的 users

  如果想要判断两列的值是否不相同,则可以使用 is distinct from:

select * from users
where has_address is distinct from has_photo
? result: 地址(address)或照片(photo)两者只有其一的用户

  not in 与 Null

  子查询(subselect)是一种很方便的过滤数据的方法。例如,如果想要查询没有任何包的用户,可以编写下面这样一个查询:

select * from users 
where id not in (select user_id from packages)

  但此时假若 packages 表中某一行的 user_id 是 null 的话,问题就来了: 返回结果是空的! 要理解为什么会发生这种古怪的事情, 我们需要理解SQL编译器究竟干了些什么. 下面是一个更简单的示例:

select * from users 
where id not in (1, 2, null)

  这个SQL语句会被转换为:

select * from users 
where id != 1 and id != 2 and id != null

  我们知道,id != null 结果是个未知值, null. 而任意值和 null 进行 and 运算的结果都是 null, 所以相当于没有其他条件. 那么出这种结果的原因就是 null 的逻辑值不为 true.

  如果条件调换过来, 查询结果就没有问题。 现在我们查询有package的用户.

select * from users 
where id in (select user_id from packages)

  同样我们可以使用简单的例子:

select * from users
where id in (1, 2, null)

  这条SQL被转换为:

select * from users 
where id = 1 or id = 2 or id = null

  因为 where 子句中是一串的 or 条件,所以其中某个的结果为 null 也是无关紧要的。非真(non-true)值并不影响子句中其他部分的计算结果,相当于被忽略了。

  Null与排序

  在排序时, null 值被认为是最大的. 在降序排序时(descending)这会让你非常头大,因为 null值排在了最前面。

  下面这个查询是为了根据得分显示用户排名, 但它将没有得分的用户排到了最前面!

select name, points
from users
order by 2 desc;

? points 为 null 的记录排在所有记录之前!

  解决这类问题有两种思路。最简单的一种是用 coalesce 消除 null的影响:

? 在输出时将 null 转换为 0 :

select name, coalesce(points, 0)
from users
order by 2 desc;

? 输出时保留 null, 但排序时转换为 0 :

select name, points
from users
order by coalesce(points, 0) desc;

  还有一种方式需要数据库的支持,指定排序时将 null 值放在最前面还是最后面:

select name, coalesce(points, 0)
from users
order by 2 desc nulls last;

  当然, null 也可以用来防止错误的发生,比如处理除数为0的数学运算错误。

  被 0 除

  除数为0是一个非常 egg-painfull 的错误。昨天还运行得好好的SQL,突然被0除一下子就出错了。一个常用的解决方法是先用 case 语句判断分母(denominator)是否为0,再进行除法运算。

select case when num_users = 0 then 0 
else total_sales/num_users end;

  ase 语句的方式其实很难看,而且分母被重复使用了。如果是简单的情况还好,如果分母是个很复杂的表达式,那么悲剧就来了: 很难读,很难维护和修改,一不小心就是一堆BUG.

  这时候我们可以看看 null 的好处. 使用 nullif 使得分母为0时变成 null. 这样就不再报错, num_users = 0 时返回结果变为 null.

select total_sales/nullif(num_users, 0);

nullif 是将其他值转为 null, 而Oracle的 nvl 是将 null 转换为其他值。

  如果不想要 null,而是希望转换为 0 或者其他数, 则可以在前一个SQL的基础上使用 coalesce函数:

select coalesce(total_sales/nullif(num_users, 0), 0);
null 再转换回0

  Conclusion

  Tony Hoare 也许会后悔自己的错误, 但至少 null 存在的问题很容易地就解决了. 那么快去练练新的大招吧,从此远离 null 挖出来的无效大坑(nullifying)!

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索sql
, null
错误
obs使用教程详细解读、边境风云剧情详细解读、弟子规与详细解读、私募基金新规详细解读、塔罗牌22张详细解读,以便于您获取更多的相关知识。

时间: 2024-10-30 12:04:20

解读SQL中的Null 超详细教程的相关文章

在开发过程中调试报表插件详细教程

原文:在开发过程中调试报表插件详细教程 背景说明 目前插件开发调试非常麻烦,需要修改代码,编译出class,重新打插件包.然后删除以前的,安装最新的.过程繁杂,而且不能调试,十分不方便.那么我就来教会大家如何可以调试插件.   需要工具:eclipse,finereport报表工具   具体说明 1. 新建工程 新建java工程,在此不赘述     2. 添加依赖,启动设计器   若要能启动设计器需要依赖的jar包很多.具体如下: A. jetty相关的jar包       B. 设计器相关的j

android自定义 iew-有没有Android中自定义布局的详细教程!

问题描述 有没有Android中自定义布局的详细教程! 有没有Android中自定义布局的详细教程!我想写一个课程表布局,在网上找了几个demo,但是里面的自定义布局看不明白! 解决方案 Composite View Composite views (也被称为 compound views) 是众多将多个view结合成为一个可重用UI组件的方法中最简单的.这种方法的实现过程是这样的: 继承相关的内建的布局. 在构造函数里面填充一个 merge 布局. 初始化成员变量并通过 findViewByI

SQL Server 2008 R2 超详细安装图文教程_mssql2008

一.下载SQL Server 2008 R2安装文件 cn_sql_server_2008_r2_enterprise_x86_x64_ia64_dvd_522233.iso 二.将安装文件刻录成光盘或者用虚拟光驱加载,或者直接解压,打开安装文件,出现下面的界面 安装SQL Server 2008 R2需要.NET Framework 3.5 SP1支持 这里我们的操作系统是Windows Server 2008 R2,已经默认自带了.NET Framework 3.5 SP1 直接点确定 选择

【超详细教程】使用Windows Live Writer 2012和Office Word 2013 发布文章到博客园全面总结

原文 http://www.cnblogs.com/liuxianan/archive/2013/04/13/3018732.html 去年就知道有这个功能,不过没去深究总结过,最近有写网络博客的欲望了,于是又重新拾起这玩意儿. 具体到底是用Windows Live Writer 2012还是用Word 2013,个人觉得看个人,因为这2个软件各有优点,各有缺点. 1.首先用LiveWriter发博客显然更专业,发布后的效果也与本地最接近,但是在编辑功能上肯定大不如Word,另外一个最大缺点是它

git超详细教程【转】

转自:http://blog.csdn.net/liuwengai/article/details/52072344 GitHub操作总结 : 总结看不明白就看下面的详细讲解.   GitHub操作流程 :   第一次提交 :   方案一 : 本地创建项目根目录, 然后与远程GitHub关联, 之后的操作一样; -- 初始化Git仓库 :Git init ; -- 提交改变到缓存 :git commit -m 'description' ; -- 本地git仓库关联GitHub仓库 : git

PhotoShop为美女照片转手绘超详细教程

原图 效果     点击查看更多>>转手绘教程 在发教程前我还是要说一句:转手绘是非常需要耐心的,许多细微的处理,往往在细节上胜出.你们往往会说:"网上的教程很多啊,可以看教程",没错,但是你们有没有发现,你跟着教程做,也不能做出他们的效果?对,即使网上的很多转手绘教程,也不能把所有过程都说出来,许多很小的处理,完全在于自己的感觉和细心.记住:没有速成的高手,任何比较出色的转手绘或是其他作品,都是靠比别人多一点点的耐心而做出来的.任何教程,都只能教你大概,而不可能教你全部.

PS鼠绘魔兽中的血精灵人物详细教程

本章最终效果如下: 使用软件: poser pro 2010(设定人物姿态)| photo shop cs5 (主要绘制) 外接设备: 鼠标 (光电)| 鼠标垫(蓝色,有个苹果哦~) 鼠绘思路: 由于没有美术基础,无法准确绘制人物透视结构,所以考虑使用poser软件设定姿态,则省去绘制线稿的过程~(画线稿这是专业人士干的事), 以前我都是使用照片内人物姿态,奈何符合要求的很少,各位同学也可以尝试使用照片作为蓝本.搞定了pos结构的问题,剩下的依靠软件和 鼠绘技术就可以全部搞定了,不管你懂不懂美术

word中输入“√”和“×” 的详细教程

  方法一:输入法输入 我们日常使用的电脑拼音输入法其实都可以快速的打出勾号和叉号,一些智能拼音输入法可以根据我们打的字来判断我们需要的符号,下面就以百度输入法为例. 输入"打勾"或者"对"时我们就可以看到有个"√"的选项. 同理我们在输入"叉"或者"错"时就会出现"×"的选项 另外我们还可以直接使用输入法的符号快捷输入v+1然后在出现的列表中找到自己要的符号. 方法二:word中使用符

wordpress中自定义菜单制作详细教程

.要想实现自定义菜单,需要用到的函数是wp_nav_menu(),给这个函数传递一些参数就可以输出自定义菜单菜单,下面简单讲讲如何使用使用这个函数.      首先,在主题目录下的functions.php的 <?php -.. ?> 之间,添加以下菜单注册代码,这样你就可以在主题文件中使用wp_nav_menu函数了:  // This theme uses wp_nav_menu() in one location. register_nav_menus();      接着我们在主题的导