原文链接:http://blog.csdn.net/wanghailong_qd/article/details/50673144

mybatis异常invalid comparison: java.util.Date and java.lang.String

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

实体类中有一个属性

  1. private Date createTime;

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

  1. create_time datetime

mapper中该属性映射的定义

  1. <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>

mybatis异常invalid comparison: java.util.Date and java.lang.String的更多相关文章

  1. mybatis部分版本异常invalid comparison: java.util.Date and java.lang.String

    严重: Servlet.service() for servlet [spring] in context with path [] threw exception [Request processi ...

  2. java.lang.IllegalArgumentException: invalid comparison: java.util.Date and java.lang.String

    在重构项目的时候,遇到了mybatis的一个异常: java.lang.IllegalArgumentException: invalid comparison: java.util.Date and ...

  3. java.util.Date、java.sql.Date、java.sql.Time、java.sql.Timestamp区别和总结

    在web开发中,避免不了对日期的操作,就几种常见的日期操作做个总结(部分参考网络,在此表示感谢): java.util.Date.java.sql.Date.java.sql.Time.java.sq ...

  4. java.util.Date、java.sql.Date、java.sql.Time、java.sql.Timestamp区别和联系

    java.util.Date.java.sql.Date.java.sql.Time.java.sql.Timestamp区别和联系 栏目:Java基础 作者:admin 日期:2015-04-19  ...

  5. java.util.Date与java.sql.Date

    我数据库里用到了日期类型.用java编程的时候同时import了java.util.*和java.sql.*,发现直接申明Date类型 Date dt; 会报错,查了一下才发现有java.util.D ...

  6. java.util.Date和java.sql.Date的区别和相互转化

    java.util.Date是在除了SQL语句的情况下面使用的.java.sql.Date是针对SQL语句使用的,它只包含日期而没有时间部分它 们都有getTime方法返回毫秒数,自然就可以直接构建. ...

  7. java.util.Date和java.sql.Date的区别和相互转化(转)

    java.util.Date是在除了SQL语句的情况下面使用的.java.sql.Date是针对SQL语句使用的,它只包含日期而没有时间部分它们都有getTime方法返回毫秒数,自然就可以直接构建.  ...

  8. java.util.Date和java.sql.Date的区别及应用

    java.util.Date 就是在除了SQL语句的情况下面使用java.sql.Date 是针对SQL语句使用的,它只包含日期而没有时间部分它都有getTime方法返回毫秒数,自然就可以直接构建ja ...

  9. java.util.Date和java.sql.Date

    java.util.Date是在除了SQL语句的情况下面使用的. java.sql.Date是针对SQL语句使用的,它只包含日期而没有时间部分 它们都有getTime方法返回毫秒数,自然就可以直接构建 ...

随机推荐

  1. 如何唯一确定一个 Java 类?

    今天偶然想起之前和朋友讨论过的一个问题:如何唯一确定一个 Java 类?我相信大多数朋友遇到这个问题的回答都是:类的全路径呗.但事实上,唯一确定一个 Java 类,单单靠类路径是不够的,还要多加上一个 ...

  2. SpringBoot中使用Quartz笔记

    Quartz可以用来做什么? Quartz是一个任务调度框架,可用来做定时任务. 吧啦吧啦......... 还是直接上代码. application.properties  配置文件. * * ? ...

  3. 蓝牙学习笔记三(Android Debug)

    android 端可以通过两种方式去Debug: 一.在手机的设置功能里,开发者模式 Enable,如下图:   http://blog.bluetooth.com/debugging-bluetoo ...

  4. Js_闭包跟作用域

    作用域的嵌套将形成作用域链,函数的嵌套将形成闭包.闭包与作用域链是 JavaScript 区别于其它语言的重要特性之一. 作用域 JavaScript 中有两种作用域:函数作用域和全局作用域. 在一个 ...

  5. 关于使用单片机读取外部电压ADC阻抗匹配的问题

    单片机的基准电压一般为3.3V,如果外部信号超过了AD测量范围,可以采用电阻分压的方法,但是要注意阻抗匹配问题.比如,SMT32的模数输入阻抗约为10K,如果外接的分压电阻无法远小于该阻值,则会因为信 ...

  6. Estimation And Gain

    Estimation: Almost every is spent on ergod the text and build the dictionary. Gains: I have never us ...

  7. Daily Scrumming* 2015.12.12(Day 4)

    一.团队scrum meeting照片 二.今日总结 姓名 WorkItem ID 工作内容 签入链接以及备注说明  江昊 任务1036 进行界面开发,明日准备开发第一个界面,社团展示界面 工作暂未完 ...

  8. BFS和DFS算法

    昨晚刚昨晚华为笔试题,用到了BFS和DFS,可惜自己学艺不精,忘记了实现原理,现在借用大佬写的内容给自己做个提高 转自:https://www.jianshu.com/p/70952b51f0c8 图 ...

  9. Supervised Hashing with Kernels, KSH

    Notation 该论文中应用到较多符号,为避免混淆,在此进行解释: n:原始数据集的大小 l:实验中用于监督学习的数据集大小(矩阵S行/列的大小) m:辅助数据集,用于得到基于核的哈希函数 r:比特 ...

  10. RYU 灭龙战 fourth day (2)

    RYU 灭龙战 fourth day (2) 前言 之前试过在ODL调用他们的rest api,一直想自己写一个基于ODL的rest api,结果还是无果而终.这个小目标却在RYU身上实现了.今日说法 ...