iBatis+Oracle+hashmap的TIMESTAMP转型问题

这是个什么问题呢?

首先你要在这样的环境下:
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中配一下:
<!--<br><br>Code highlighting produced by Actipro CodeHighlighter (freeware)<br>http://www.CodeHighlighter.com/<br><br>--> <? xml version="1.0" encoding="UTF-8" ?>
<! DOCTYPE sqlMapConfig PUBLIC "-//iBATIS.com//DTD SQL Map Config 2.0//EN"
    "http://www.ibatis.com/dtd/sql-map-config-2.dtd"
>
< sqlMapConfig >
    
< settings  dot.gif />
    
< typeHandler  javaType ="object"  
        callback
="xxx.xxx.xxx.OracleObjectTypeHandler" />
    
    
< sqlMap   dot.gif />
    dot.gif
</ sqlMapConfig >

这样就OK了!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值