这是个什么问题呢?
首先你要在这样的环境下:
iBatis-2.3.4.726
oracle jdbc driver 10.2.0.3.0
你的数据库中字段定义为TIMESTAMP
你的sqlMap中的resultClass为hashmap
这是如果你这样调用
Map result = xxxDao.queryForObject(......);
Date someTime = (Date)result.get("someTime");
你会的到如下ClassCaseException: oracle.sql.TIMESTAMP
因为oracle返回的是其自己的TIMESTAMP类型的对象, 而弱智的iBatis, 遇到resultClass为map及其子类型时, 狼心狗肺的, 把数据的原始类型原样返回. 这是是你的到的map里面都是些什么 oracle.sql.DATE, oracle.sql.TIMESTAMP什么的.
怎么办呢?
难道要这样?
Date someTime = ((oracle.sql.TIMESTAMP)result.get("someTime")).dateValue(); // <- 这还是java.sql.Date
实在是太ugly了!
其实iBatis在遇到map类型的ResultClass时, 永远使用com.ibatis.sqlmap.engine.type.ObjectTypeHandler来做类型转换, 而这个类里的实现却又是, 永远返回JDBC驱动自己的类型.
这样只要我们自己写一个ObjectTypeHandler来代替这个该死的东西不就完事了吗!
于是乎~~~~
如下:
<!--<br><br>Code highlighting produced by Actipro CodeHighlighter (freeware)<br>http://www.CodeHighlighter.com/<br><br>-->
/**
* The TypeHandler for Oracle JDBC Driver, which fix the
* "oracle.sql.TIMESTAMP ClassCastException" problem
*
*
@author
matianyi
*
*/
public
class
OracleObjectTypeHandler
extends
BaseTypeHandler
implements
TypeHandler {
/**
*
@see
TypeHandler#setParameter(PreparedStatement, int, Object, String)
*/
public
void
setParameter(PreparedStatement ps,
int
i, Object parameter, String jdbcType)
throws
SQLException {
ps.setObject(i, parameter);
}
/**
*
@see
TypeHandler#getResult(ResultSet, String)
*/
public
Object getResult(ResultSet rs, String columnName)
throws
SQLException {
Object object
=
rs.getObject(columnName);
if
(rs.wasNull()) {
return
null
;
}
else
{
return
fix(object);
}
}
/**
*
@see
TypeHandler#getResult(ResultSet, int)
*/
public
Object getResult(ResultSet rs,
int
columnIndex)
throws
SQLException {
Object object
=
rs.getObject(columnIndex);
if
(rs.wasNull()) {
return
null
;
}
else
{
return
fix(object);
}
}
/**
*
@see
TypeHandler#getResult(CallableStatement, int)
*/
public
Object getResult(CallableStatement cs,
int
columnIndex)
throws
SQLException {
Object object
=
cs.getObject(columnIndex);
if
(cs.wasNull()) {
return
null
;
}
else
{
return
fix(object);
}
}
/**
*
@see
TypeHandler#valueOf(String)
*/
public
Object valueOf(String s) {
return
s;
}
/**
* !@#@%$%^$%%!@#%$%^#$%#!%#$^$#%^
*
*
@param
obj
* object from oracle jdbc driver
*
@return
object of suitable java datatype
*/
protected
Object fix(Object obj) {
try
{
if
(obj
instanceof
TIMESTAMP) {
return
new
Date(((TIMESTAMP) obj).dateValue().getTime());
}
else
if
(obj
instanceof
DATE) {
return
new
Date(((DATE) obj).dateValue().getTime());
}
else
if
(obj
instanceof
TIMESTAMPLTZ) {
return
new
Date(((TIMESTAMPLTZ) obj).dateValue().getTime());
}
else
if
(obj
instanceof
TIMESTAMPTZ) {
return
new
Date(((TIMESTAMPTZ) obj).dateValue().getTime());
}
else
{
return
obj;
}
}
catch
(Exception e) {
return
obj;
}
}
}
上面这个相信大家都回写! 然后, 关键的来了: 怎么用?
在sqlMapConfig中配一下:
这样就OK了!