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

開發中改動mapper文件後須要從新編譯發佈, 因爲工程比較大很是耗時, 因此爲方便快速測試乾脆寫了一個小Java工程. 工程中用到的dao, mapper和實體類都是從工程中拷出來的, 數據庫也是同一個. 可是遇到一個比較奇怪的問題java

 

實體類中有一個屬性mysql

 

private Date createTime;

  

 

對應該屬性數據庫中定義的是sql

 

create_time datetime

  

mapper中該屬性映射的定義數據庫

 

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

  

如下是mapper中對應Dao方法SQL語句apache

 

<select id="selectByCreateTime" resultMap="userMap">
  select * from user 
  <where>
	<if test="createTime != null and createTime !='' " >
	  date(create_time) = date(#{createTime,jdbcType=TIMESTAMP})
	</if>
  </where>
</select>

  

 

其中date()函數只是用來把年月日時分秒的日期截取爲年月日, 這個對於該異常沒有任何影響

在測試類中建立實體併爲其屬性賦值session

 

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

  

而後執行查詢方法dao.selectByCreateTime(user)的時候就報錯了mybatis

Exception in thread "main" org.apache.ibatis.exceptions.PersistenceException: 
### Error querying database.  Cause: java.lang.IllegalArgumentException: invalid comparison: java.util.Date and java.lang.String
### Cause: java.lang.IllegalArgumentException: invalid comparison: java.util.Date and java.lang.String
	at org.apache.ibatis.exceptions.ExceptionFactory.wrapException(ExceptionFactory.java:30)
	at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:122)
	at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:113)
	at org.apache.ibatis.binding.MapperMethod.executeForMany(MapperMethod.java:122)
	at org.apache.ibatis.binding.MapperMethod.execute(MapperMethod.java:64)
	at org.apache.ibatis.binding.MapperProxy.invoke(MapperProxy.java:53)
	at com.sun.proxy.$Proxy0.selectByCreateTime(Unknown Source)
	at mybatis.Test.buyerInfoTimeTest(Test.java:53)
	at mybatis.Test.main(Test.java:39)
Caused by: java.lang.IllegalArgumentException: invalid comparison: java.util.Date and java.lang.String
	at org.apache.ibatis.ognl.OgnlOps.compareWithConversion(OgnlOps.java:92)
	at org.apache.ibatis.ognl.OgnlOps.isEqual(OgnlOps.java:142)
	at org.apache.ibatis.ognl.OgnlOps.equal(OgnlOps.java:794)
	at org.apache.ibatis.ognl.ASTNotEq.getValueBody(ASTNotEq.java:53)
	at org.apache.ibatis.ognl.SimpleNode.evaluateGetValueBody(SimpleNode.java:212)
	at org.apache.ibatis.ognl.SimpleNode.getValue(SimpleNode.java:258)
	at org.apache.ibatis.ognl.ASTAnd.getValueBody(ASTAnd.java:61)
	at org.apache.ibatis.ognl.SimpleNode.evaluateGetValueBody(SimpleNode.java:212)
	at org.apache.ibatis.ognl.SimpleNode.getValue(SimpleNode.java:258)
	at org.apache.ibatis.ognl.Ognl.getValue(Ognl.java:494)
	at org.apache.ibatis.ognl.Ognl.getValue(Ognl.java:458)
	at org.apache.ibatis.scripting.xmltags.OgnlCache.getValue(OgnlCache.java:44)
	at org.apache.ibatis.scripting.xmltags.ExpressionEvaluator.evaluateBoolean(ExpressionEvaluator.java:32)
	at org.apache.ibatis.scripting.xmltags.IfSqlNode.apply(IfSqlNode.java:34)
	at org.apache.ibatis.scripting.xmltags.MixedSqlNode.apply(MixedSqlNode.java:33)
	at org.apache.ibatis.scripting.xmltags.TrimSqlNode.apply(TrimSqlNode.java:55)
	at org.apache.ibatis.scripting.xmltags.MixedSqlNode.apply(MixedSqlNode.java:33)
	at org.apache.ibatis.scripting.xmltags.DynamicSqlSource.getBoundSql(DynamicSqlSource.java:41)
	at org.apache.ibatis.mapping.MappedStatement.getBoundSql(MappedStatement.java:280)
	at org.apache.ibatis.executor.CachingExecutor.query(CachingExecutor.java:80)
	at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:120)
	... 7 more

  

看樣子是由於類型不符合, 可是想了想, Date類型對應MySQL的datetime, 以及mapper中jdbcType都沒問題啊. 並且徹底同樣的東西在原工程中是徹底正常的. 既然都是同樣的代碼, 那就找找倆工程有啥不同的吧app

 

 

首先是MySQL jar版本不一樣. 換成原工程中的版本也無效. 而後是mybatis jar版本不同, 換成原工程中的版本問題就解決了!函數

 

原工程中配置的是mybatis-3.2.8, 而我測試工程中用的是mybatis-3.3.0.後來在網上找了一下才知道, 原來這是mybatis 3.3.0中對於時間參數進行比較時的一個bug. 若是拿傳入的時間類型參數與空字符串''進行對比判斷則會引起異常. 因此在上面的代碼中去該該判斷, 只保留非空判斷就正常了測試

 

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

  

 

<if test="createTime != null">
  date(create_time) = date(#{createTime,jdbcType=TIMESTAMP})
</if>
相關文章
相關標籤/搜索