Mybatis在进行<if test="status != null and status != ''">判空操作时,假设status为0的时候,该推断条件的值为false,也就是说Mybatis此时把0作为null来进行推断的,所以遇到这样的情况时。仅仅能依照以下这种方法来办!

一、源代码调查结果

遇到这个问题时。我翻看了Mybatis的源代码,发现其在ExpressionEvaluator.java类的evaluateBoolean方法处返回了false,这个源代码包的分析方式和源代码分析 There is no getter for property named ‘*’ in ‘class java.lang.String有极大的同样之处。

但发现了问题。也然并卵,改动Mybatis的源代码我是不行,所以仅仅能曲线救国!

二、场景分析

数据库字段

  `status` tinyint(2) NOT NULL COMMENT '0未支付,1已支付',

此时0表示未支付,1表示支付。那么依照一般的逻辑来看。查询语句须要写成这样的形式:

select * from ym_deals y WHERE d.deal_id = 1

        <if test="status != null and status != ''">
and y.status = #{status}
</if>

其意义在于。当查询參数不传递未支付或者支付条件时。就须要查询status为0和1的所有结果。也就是说没有条件and y.status = #{status}。但事实就纠结了,当status传递的參数为1时,if条件的结果为true,也就是说sql语句变成了select * from ym_deals y WHERE d.deal_id = 1 and y.status=1,查询结果正常,但假设status为0此时,sql语句变成了select * from ym_deals y WHERE d.deal_id = 1,与预期结果select * from ym_deals y WHERE d.deal_id = 1 and y.status=0显然是不符的!

也就是说,Mybatis自作主张的把status为0的条件当做了if的false结果!

三、解决的方法

①、对传递參数拦截

    public static int parseStringToInt(Object parameter, int defualtValue) {
try {
if ((parameter == null) || (parameter.equals(""))) {
return defualtValue;
}
return Integer.parseInt(parameter.toString());
} catch (Exception ex) {
}
return 0;
}
vo.addParams("status", StrUtil.parseStringToInt(vo.getParams("status"), -1));

也就是说在查询所有支付状态的订单时,把status的值转换为-1.

②、重构Mybatis的查询语句

select * from ym_deals y WHERE d.deal_id = 1

        <if test="status != -1">
and y.status = #{status}
</if>

此时不再判空,而判-1。

依照以上的办法就攻克了问题。不知道你那是否有更好的解决的方法?


四、后记

这篇文章发表之后,有热心的朋友给出以下的解决的方法:

lengjian68 status是Integer 类型的 把status!=‘’ 给去掉

去掉就好了

1楼 qq_30159115 确定是 int类型的就不用 推断 status != ‘’ 就搞定了

Re: Optimistic_ 发表 回复qq_30159115:给力啊。感谢!

推断不是为空。对于int类型就是在推断不为0么?

依照朋友给出的观点。我尝试了一下,结果如朋友所说。status确定为integer类型的,把status!=”给去掉就好了。

那么如今又一次来看,我本来的做法就有问题,integer类型的,我干嘛要用status!=”来进行推断,这显然是一种不负责任的做法,至于<if test="status != null and status != ''">的推断条件。显然是错误的,从原则上来讲,该推断条件仅仅能来推断status为string类型的,用来推断integer类型。岂不贻笑慷慨,至于后面我提出的解决方式,显然也就无力苍白。我仅仅想把这篇文章立刻删掉,好不再误导他人。

但看到

Re: Optimistic_ 发表 回复qq_30159115:给力啊!感谢!

推断不是为空,对于int类型就是在推断不为0么?

以下这位朋友的疑问,我认为,这篇文章存在另一些价值,由于还有和我一样的朋友在犯错。“推断不是为空,对于int类型就是在推断不为0么?”这样的认识显然是第二种错误,integer类型非常明显存在为null的情况,那么对于int类型当然不是在推断为0的。(当然了。这位仁兄说int类型,在严格意义上讲。就不存在null的可能了,那么我仅仅能武断的猜測任务。这位朋友是在说integer,请原谅我的鲁莽)

那么得出下面结论:

  1. <if test="status != null">中status为integer类型的。status=0的推断结果为true。
  2. <if test="status != null and status != ''">中status为integer类型的,status=0的推断结果为false。mybatis把status作为了string来进行推断。

  3. 假设status为integer,提前对status进行去空操作,再推断<if test="status != -1">在有的时候也是一种曲线救国。

相关文章


感谢您阅读【沉默王二的博客】,假设王二的博客给您带来一丝帮助或感动。我(也就是王二)将不甚荣幸。

假设您碰巧喜欢。能够留言或者私信我,这将是我鼓捣很多其它优秀文章的最强动力。

Mybatis+0+null,小问题引发的血案的更多相关文章

  1. elasticsearch5.0.1集群一次误删除kibana索引引发的血案

    elasticsearch集群中一次删除kibana索引引发的血案 1.问题发生的过程: 早上的时候有某个索引无法看到报表数据,于是就点该报表多次,估计集群被点挂了,报错:Elasticsearch ...

  2. 一个无锁消息队列引发的血案(六)——RingQueue(中) 休眠的艺术 [续]

    目录 (一)起因 (二)混合自旋锁 (三)q3.h 与 RingBuffer (四)RingQueue(上) 自旋锁 (五)RingQueue(中) 休眠的艺术 (六)RingQueue(中) 休眠的 ...

  3. 一个无锁消息队列引发的血案(五)——RingQueue(中) 休眠的艺术

    目录 (一)起因 (二)混合自旋锁 (三)q3.h 与 RingBuffer (四)RingQueue(上) 自旋锁 (五)RingQueue(中) 休眠的艺术 (六)RingQueue(中) 休眠的 ...

  4. 一个无锁消息队列引发的血案(三)——地:q3.h 与 RingBuffer

    目录 (一)起因 (二)混合自旋锁 (三)q3.h 与 RingBuffer (四)RingQueue(上) 自旋锁 (五)RingQueue(中) 休眠的艺术 (六)RingQueue(中) 休眠的 ...

  5. Integer.parseInt 引发的血案

    Integer.parseInt 处理一个空字符串, 结果出错了, 程序没有注意到,搞了很久, 引发了血案啊!! 最后,终于 观察到了, 最后的部分: Caused by: java.lang.NoC ...

  6. Replication的犄角旮旯(六)-- 一个DDL引发的血案(上)(如何近似估算DDL操作进度)

    <Replication的犄角旮旯>系列导读 Replication的犄角旮旯(一)--变更订阅端表名的应用场景 Replication的犄角旮旯(二)--寻找订阅端丢失的记录 Repli ...

  7. 转:一个Sqrt函数引发的血案

    转自:http://www.cnblogs.com/pkuoliver/archive/2010/10/06/1844725.html 源码下载地址:http://diducoder.com/sotr ...

  8. 一个Sqrt函数引发的血案(转)

    作者: 码农1946  来源: 博客园  发布时间: 2013-10-09 11:37  阅读: 4556 次  推荐: 41   原文链接   [收藏]   好吧,我承认我标题党了,不过既然你来了, ...

  9. 一个无锁消息队列引发的血案(四)——月:RingQueue(上) 自旋锁

    目录 (一)起因 (二)混合自旋锁 (三)q3.h 与 RingBuffer (四)RingQueue(上) 自旋锁 (五)RingQueue(中) 休眠的艺术 (六)RingQueue(中) 休眠的 ...

随机推荐

  1. JavaScript--改变 HTML 样式

    HTML DOM 允许 JavaScript 改变 HTML 元素的样式.如何改变 HTML 元素的样式呢? 语法: Object.style.property=new style; 注意:Objec ...

  2. redis cluster集群理解

    Redis Cluster集群 一.redis-cluster设计 Redis集群搭建的方式有多种,例如使用zookeeper等,但从redis 3.0之后版本支持redis-cluster集群,Re ...

  3. 大数据~说说ZooKeeper

    一些概念 ZooKeeper是一个分布式的,开放源码的分布式应用程序协调服务,是Google的Chubby一个开源的实现,是Hadoop和Hbase和Kafka重要组件.它是一个为分布式应用提供一致性 ...

  4. Floyd模板

    比较简单的算法:但是当点太多需要剪枝,不然很耗时 void Floyd() { ;k<n;++k) ;i<n;++i) ;j<n;++j) dj[i][j] = min(dj[i][ ...

  5. python中struct.pack()函数和struct.unpack()函数

    python中的struct主要是用来处理C结构数据的,读入时先转换为Python的字符串类型,然后再转换为Python的结构化类型,比如元组(tuple)啥的~.一般输入的渠道来源于文件或者网络的二 ...

  6. js分页插件

    //分页插件1function showView(option) {    //参数定义id,页容量,当前页,总数,页总数    var id = option.id,         pageSiz ...

  7. ECharts实例开发学习笔记二——时间轴

    记录一下ECharts时间轴的使用,首先是照着官方的示例做一遍,在这里不要忘了引入timelineOption.js,后面介绍如何动态创建时间轴的记录数,即根据需求可伸缩显示有多少天或者年月等轴标记数 ...

  8. HDU_1026_Ignatius and the Princess I_BFS(保存路径)

    Ignatius and the Princess I Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (J ...

  9. sql日期提取

    --插入数据修改不行:必须提供学号 insert into Student(生日类型) values('阳历') --把月份提取出来 显示两位数 select DATENAME(month,getda ...

  10. PC、h5项目接入第三方支付宝扫码登录、扫码付款

    首先介绍一下pc项目接入支付宝扫码支付. 1.pc.移动接入支付宝扫码支付. 其实这个逻辑很简单,前端所需要处理的不是很多,后台会给一个连接,前端只需要将要支付的订单id拼接在这个连接上,然后打开跳转 ...