<select id="getReceivableStatisticsList" parameterType="com.unicom.apartment.bo.ReceivableStatisticsTopParmBo" resultMap="BaseResultMap3">
        select  a.*   , b.id as money_register_id  ,b.bill_id as bill_id      ,b.register_type as register_type   ,b.money as money   , b.is_deleted as register_type_delete
        FROM  bill AS a LEFT JOIN  money_register AS b ON   a.id  =  b.bill_id
        WHERE a.is_deleted = 'N'
        <if test="billTypeList != null" >
            and bill_type in
            <!-- 这里要将collection属性的值标记为list,不然不知道传入的参数是list separator表示分离器 item表示list中的一个元素 -->
            <foreach item="billTypeItem" index="index" collection="billTypeList" open="("
                     separator="," close=")"> #{billTypeItem}</foreach>
        </if>
        <if test="resourceIdList != null" >
            and resource_id in
            <!-- 这里要将collection属性的值标记为list,不然不知道传入的参数是list separator表示分离器 item表示list中的一个元素 -->
            <foreach item="resourceIdItem" index="index" collection="resourceIdList" open="("
                     separator="," close=")"> #{resourceIdItem}</foreach>
        </if>
        <if test="startTime != null" >
            and  date(a.bill_finaldate) BETWEEN date(#{startTime,jdbcType=TIMESTAMP}) and date(#{endTime,jdbcType=TIMESTAMP})
        </if>
</select>


mybaits3.3.0date类型数据判断非空报错

转载永远的白鼠 最后发布于2018-07-17 17:10:47 阅读数 1148  收藏

开发中改动mapper文件后需要重新编译发布, 由于工程比较大非常耗时, 所以为方便快速测试干脆写了一个小java工程. 工程中用到的dao, mapper和实体类都是从工程中拷出来的, 数据库也是同一个. 但是遇到一个比较奇怪的问题

 

实体类中有一个属性

 

private Date createTime;

 

对应该属性数据库中定义的是

 

create_time datetime

 

 

 

mapper中该属性映射的定义

 

<result column="create_time" property="createTime" jdbcType="TIMESTAMP" />

 

 

 

以下是mapper中对应Dao方法SQL语句

 


 

1. <select id="selectByCreateTime" resultMap="userMap">
2. select * from user
3. <where>
4. <if test="createTime != null and createTime !='' " >
5. date(create_time) = date(#{createTime,jdbcType=TIMESTAMP})
6. </if>
7. </where>
8. </select>

其中date()函数只是用来把年月日时分秒的日期截取为年月日, 这个对于该异常没有任何影响

在测试类中创建实体并为其属性赋值

 


 

1. User user=new User();
2. user.setCreateTime(new SimpleDateFormat("yyyy-MM-dd").parse("2016-01-18"));

然后执行查询方法dao.selectByCreateTime(user)的时候就报错了

 


 

1. Exception in thread "main" org.apache.ibatis.exceptions.PersistenceException:
2. ### Error querying database. Cause: java.lang.IllegalArgumentException: invalid comparison: java.util.Date and java.lang.String
3. ### Cause: java.lang.IllegalArgumentException: invalid comparison: java.util.Date and java.lang.String
4. at org.apache.ibatis.exceptions.ExceptionFactory.wrapException(ExceptionFactory.java:30)
5. at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:122)
6. at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:113)
7. at org.apache.ibatis.binding.MapperMethod.executeForMany(MapperMethod.java:122)
8. at org.apache.ibatis.binding.MapperMethod.execute(MapperMethod.java:64)
9. at org.apache.ibatis.binding.MapperProxy.invoke(MapperProxy.java:53)
10. at com.sun.proxy.$Proxy0.selectByCreateTime(Unknown Source)
11. at mybatis.Test.buyerInfoTimeTest(Test.java:53)
12. at mybatis.Test.main(Test.java:39)
13. Caused by: java.lang.IllegalArgumentException: invalid comparison: java.util.Date and java.lang.String
14. at org.apache.ibatis.ognl.OgnlOps.compareWithConversion(OgnlOps.java:92)
15. at org.apache.ibatis.ognl.OgnlOps.isEqual(OgnlOps.java:142)
16. at org.apache.ibatis.ognl.OgnlOps.equal(OgnlOps.java:794)
17. at org.apache.ibatis.ognl.ASTNotEq.getValueBody(ASTNotEq.java:53)
18. at org.apache.ibatis.ognl.SimpleNode.evaluateGetValueBody(SimpleNode.java:212)
19. at org.apache.ibatis.ognl.SimpleNode.getValue(SimpleNode.java:258)
20. at org.apache.ibatis.ognl.ASTAnd.getValueBody(ASTAnd.java:61)
21. at org.apache.ibatis.ognl.SimpleNode.evaluateGetValueBody(SimpleNode.java:212)
22. at org.apache.ibatis.ognl.SimpleNode.getValue(SimpleNode.java:258)
23. at org.apache.ibatis.ognl.Ognl.getValue(Ognl.java:494)
24. at org.apache.ibatis.ognl.Ognl.getValue(Ognl.java:458)
25. at org.apache.ibatis.scripting.xmltags.OgnlCache.getValue(OgnlCache.java:44)
26. at org.apache.ibatis.scripting.xmltags.ExpressionEvaluator.evaluateBoolean(ExpressionEvaluator.java:32)
27. at org.apache.ibatis.scripting.xmltags.IfSqlNode.apply(IfSqlNode.java:34)
28. at org.apache.ibatis.scripting.xmltags.MixedSqlNode.apply(MixedSqlNode.java:33)
29. at org.apache.ibatis.scripting.xmltags.TrimSqlNode.apply(TrimSqlNode.java:55)
30. at org.apache.ibatis.scripting.xmltags.MixedSqlNode.apply(MixedSqlNode.java:33)
31. at org.apache.ibatis.scripting.xmltags.DynamicSqlSource.getBoundSql(DynamicSqlSource.java:41)
32. at org.apache.ibatis.mapping.MappedStatement.getBoundSql(MappedStatement.java:280)
33. at org.apache.ibatis.executor.CachingExecutor.query(CachingExecutor.java:80)
34. at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:120)
35. ... 7 more

看样子是因为类型不符合, 但是想了想, Date类型对应MySQL的datetime, 以及mapper中jdbcType都没问题啊. 而且完全一样的东西在原工程中是完全正常的. 既然都是一样的代码, 那就找找俩工程有啥不一样的吧

 

 

首先是mysql jar版本不同. 换成原工程中的版本也无效. 然后是mybatis jar版本不一样, 换成原工程中的版本问题就解决了!

 

原工程中配置的是mybatis-3.2.8, 而我测试工程中用的是mybatis-3.3.0.后来在网上找了一下才知道, 原来这是mybatis 3.3.0中对于时间参数进行比较时的一个bug. 如果拿传入的时间类型参数与空字符串''进行对比判断则会引发异常. 所以在上面的代码中去该该判断, 只保留非空判断就正常了

 

 


 

1. <if test="createTime != null and createTime !='' " >
2. date(create_time) = date(#{createTime,jdbcType=TIMESTAMP})
3. </if>


 


 

改为

 

1. <if test="createTime != null">
2. date(create_time) = date(#{createTime,jdbcType=TIMESTAMP})
3. </if>