mybatis-自定义TypeHandler

步骤1:实现TypeHandler接口

import java.sql.CallableStatement;
import java.sql.PreparedStatement;    
import java.sql.ResultSet;    
import java.sql.SQLException;    
    
import org.apache.ibatis.type.JdbcType;
import org.apache.ibatis.type.MappedJdbcTypes;
import org.apache.ibatis.type.MappedTypes;
import org.apache.ibatis.type.TypeHandler;
    
/**  
 * @author   
 * java中的boolean和jdbc中的char之间转换;true-Y;false-N  
 */
//方式一:
@MappedJdbcTypes(JdbcType.CHAR)
@MappedTypes(Boolean.class)
public class BooleanTypeHandler implements TypeHandler {    
    
    /* (non-Javadoc)  
     * @see org.apache.ibatis.type.TypeHandler#getResult(java.sql.ResultSet, java.lang.String)  
     */    
    @Override    
    public Object getResult(ResultSet resultSet, String columnLabel) throws SQLException {
        String str = resultSet.getString(columnLabel);
        Boolean rt = Boolean.FALSE;    
        if (str.equalsIgnoreCase("Y")){
            rt = Boolean.TRUE;    
        }    
        return rt;     
    }

    @Override
    public Object getResult(ResultSet resultSet, int columnIndex) throws SQLException {
        String str = resultSet.getString(columnIndex);
        Boolean rt = Boolean.FALSE;
        if (str.equalsIgnoreCase("Y")){
            rt = Boolean.TRUE;
        }
        return rt;
    }

    /* (non-Javadoc)  
     * @see org.apache.ibatis.type.TypeHandler#getResult(java.sql.CallableStatement, int)  
     */    
    @Override    
    public Object getResult(CallableStatement arg0, int arg1)    
            throws SQLException {    
        Boolean b = arg0.getBoolean(arg1);    
        return b == true ? "Y" : "N";    
    }    
    
    /* (non-Javadoc)  
     * @see org.apache.ibatis.type.TypeHandler#setParameter(java.sql.PreparedStatement, int, java.lang.Object, org.apache.ibatis.type.JdbcType)  
     */    
    @Override    
    public void setParameter(PreparedStatement arg0, int arg1, Object arg2,    
            JdbcType arg3) throws SQLException {    
        Boolean b = (Boolean) arg2;    
        String value = (Boolean) b == true ? "Y" : "N";    
        arg0.setString(arg1, value);    
    }    
}    

步骤2:在Mybatis配置中注册该TypeHandler

<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
        <property name="dataSource" ref="dataSource" />
        <property name="mapperLocations" value="classpath:dao/mapper/*.xml"></property>
        <property name="typeHandlers" >
            <array>
                <bean class="com.sankuai.travel.csc.test.dao.typehandler.BooleanTypeHandler" />
            </array>
        </property>
        <property name="plugins">
            <array>
                <bean class="com.sankuai.travel.csc.test.dao.interceptor.SqlMonitorInterceptor" />
            </array>
        </property>
    </bean>

步骤3:在映射配置文件中使用该TypeHander(如果第一步使用了注解,此处可以省略)

3.1在resultMap的定义中对对应列定义typeHandler

<resultMap id="BaseResultMap" type="com.sankuai.travel.csc.test.dao.domain.CategoryDO" >
    <id column="id" property="id" jdbcType="INTEGER" />
    <result column="category_type" property="categoryType" jdbcType="INTEGER" />
    <result column="category_name" property="categoryName" jdbcType="VARCHAR" />
    <result column="allocate_time" property="allocateTime" jdbcType="INTEGER" />
    <result column="promise_time" property="promiseTime" jdbcType="INTEGER" />
    <result column="last_operator" property="lastOperator" jdbcType="VARCHAR" />
    <result column="update_time" property="updateTime" jdbcType="TIMESTAMP" />

    <!--写在这里,仅对select语句有效,typeHandler="com.sankuai.travel.csc.test.dao.typehandler.BooleanTypeHandler" -->
    <result column="is_cancel" property="isCancel" jdbcType="CHAR" typeHandler="com.sankuai.travel.csc.test.dao.typehandler.BooleanTypeHandler"/>

    <result column="status1" property="status1" jdbcType="CHAR" />
    <result column="status2" property="status2" jdbcType="CHAR" />
    <result column="phone_list" property="phoneList" jdbcType="VARCHAR" />
    <result column="create_time" property="createTime" jdbcType="TIMESTAMP" />
  </resultMap>

这里只能是在select的时候才会使用自定义的TypeHandler处理对应的映射关系,如果要在insert或者update时使用则需要在sql定义中添加相应的内容

3.2在resultMap的定义中对对应列定义typeHandler

<insert id="insert" parameterType="com.sankuai.travel.csc.test.dao.domain.CategoryDO" >
    <selectKey resultType="java.lang.Integer" keyProperty="id" order="AFTER" >
      SELECT LAST_INSERT_ID()
    </selectKey>
    insert into hotel_order_category (category_type, category_name, allocate_time, 
      promise_time, last_operator, update_time, 
      is_cancel, status1, status2, 
      phone_list, create_time)
    values (#{categoryType,jdbcType=INTEGER}, #{categoryName,jdbcType=VARCHAR}, #{allocateTime,jdbcType=INTEGER}, 
      #{promiseTime,jdbcType=INTEGER}, #{lastOperator,jdbcType=VARCHAR}, #{updateTime,jdbcType=TIMESTAMP}, 
      <!--写在这里,仅对insert有效,注意,没有双引号: typeHandler=com.sankuai.travel.csc.test.dao.typehandler.BooleanTypeHandler -->
      #{isCancel,jdbcType=CHAR,typeHandler=com.sankuai.travel.csc.test.dao.typehandler.BooleanTypeHandler},
      #{status1,jdbcType=CHAR}, #{status2,jdbcType=CHAR},
      #{phoneList,jdbcType=VARCHAR}, #{createTime,jdbcType=TIMESTAMP})
  </insert>

#mybatis-enumtypehandler和enumordinaltypehandler的区别

mybatis对枚举类型提供了两种类型支持:EnumTypeHandler和EnumOrdinalTypeHandler。

  • EnumTypeHandler是mybatis默认的枚举类型转换器,如果pojo类中使用了枚举类型,而配置文件没有指定类型转换类,mybatis将使用EnumTypeHandler处理枚举属性。EnumTypeHandler的将把枚举类的name进行存储,枚举类的name即枚举类名。参考:http://mybatis.github.io/mybatis-3/zh/configuration.html
  • EnumOrdinalTypeHandler是mybatis提供的另一种转换器,顾名思义这个转换类使用了枚举类的ordinal属性作为数据库存储信息,由于ordinal属性是int类型的,按照官网的说明数据库中对应资源应该是int或double类型的,但是个人测试过程中MYSQL的varchar字段也可以存储。
  • 总结:EnumTypeHandler和EnumOrdinalTypeHandler的区别主要是数据库中存储字段的类型差别,由于EnumOrdinalTypeHandler使用枚举类型的ordinal作为存储,所以必须使用数字类型字段存储。
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
        <property name="dataSource" ref="dataSource" />
        <property name="mapperLocations" value="classpath:dao/mapper/*.xml"></property>
        <property name="typeHandlers" >
            <array>
                <bean class="com.sankuai.travel.csc.test.dao.typehandler.BooleanTypeHandler" />
                <bean class="com.sankuai.travel.csc.test.dao.typehandler.StringArrayTypeHandler" />
  <!--不需要EnumTypeHandler,默认包含 -->
                <bean class="org.apache.ibatis.type.EnumTypeHandler" >
                    <constructor-arg name="type" value="com.sankuai.travel.csc.test.enums.SimpleStatusEnum" />
                </bean>
<!--需要EnumOrdinalTypeHandler,默认包含 -->
                <bean class="org.apache.ibatis.type.EnumOrdinalTypeHandler" >
                    <constructor-arg name="type" value="com.sankuai.travel.csc.test.enums.SimpleStatusEnum" />
                </bean>
            </array>
        </property>
        <property name="plugins">
            <array>
                <bean class="com.sankuai.travel.csc.test.dao.interceptor.SqlMonitorInterceptor" />
            </array>
        </property>
    </bean>
<resultMap id="BaseResultMap" type="com.sankuai.travel.csc.test.dao.domain.CategoryDO" >
    <id column="id" property="id" jdbcType="INTEGER" />
    <result column="category_type" property="categoryType" jdbcType="INTEGER" />
    <result column="category_name" property="categoryName" jdbcType="VARCHAR" />
    <result column="allocate_time" property="allocateTime" jdbcType="INTEGER" />
    <result column="promise_time" property="promiseTime" jdbcType="INTEGER" />
    <result column="last_operator" property="lastOperator" jdbcType="VARCHAR" />
    <result column="update_time" property="updateTime" jdbcType="TIMESTAMP" />

    <!--方式二:写在这里,仅对select语句有效,typeHandler="com.sankuai.travel.csc.test.dao.typehandler.BooleanTypeHandler" -->
    <result column="is_cancel" property="isCancel" jdbcType="CHAR" />
<!--EnumTypeHandler默认处理类,不需要配置 -->
    <result column="status1" property="status1" jdbcType="CHAR" />
<!--EnumOrdinalTypeHandler需要单独配置 -->
    <result column="status2" property="status2" jdbcType="CHAR" typeHandler="org.apache.ibatis.type.EnumOrdinalTypeHandler" />
    <result column="phone_list" property="phoneList" jdbcType="VARCHAR" />
    <result column="create_time" property="createTime" jdbcType="TIMESTAMP" />
  </resultMap>


参考文章

https://my.oschina.net/amoshuang/blog/134199
mybatis-enumtypehandler和enumordinaltypehandler的区别

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,222评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,455评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,720评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,568评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,696评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,879评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,028评论 3 409
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,773评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,220评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,550评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,697评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,360评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,002评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,782评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,010评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,433评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,587评论 2 350

推荐阅读更多精彩内容