有些时候,java代码的类型和数据库jdbc的类型不一致时,需要自定义类型转换器,让mybatis能够识别。

比如java代码中boolean类型,存储到数据库中时用number类型。(true--1,false--0)

此时就需要我们创建类型转换器。创建方式有两种,一种是实现TypeHandler接口,还有一种是继承BaseTypeHandler实现类。观察源码发现,baseTypeHandler是TypeHandler的实现类,所以我们使用继承BaseTypeHandler的方式实现起来更方便.

mybatais javaType 自定义 mybatis自定义handler_类型转换器

自定义一个类,BooleanAndIntConverter,重写4个方法,1个set,3个get,同时继承BaseTypeHandler时,存在泛型约束,可以指定需要转换的java类型Boolean。

mybatais javaType 自定义 mybatis自定义handler_类型转换器_02

其中setNonNullParameter表示将java的类型,set成数据库jdbc类型,而三个get方法则相反,表示从数据库中获取jdbc类型的数据,转换成java类型。set方法中4个参数分别表示:PreparedStatement对象,操作对象参数的位置,java类型(这里是Boolean类型),jdbc操作数据库类型。

@Override
     public Boolean getNullableResult(ResultSet rs, String columnName) throws SQLException {
         int sexNum = rs.getInt(columnName) ;//rs.getInt("stuno") ;
         return sexNum == 1?true:false ;
     }

重写完方法,我们就可以实现逻辑。通过ResultSet对象获取数据库中的值,然后再通过三目运算符继续判断转换即可。ResultSet对象的get方法,里面的参数可以是下标columnName,也可以是指定名字columnName。除了通过ResultSet拿值,也可以通过存储过程拿,方法都一样。

mybatais javaType 自定义 mybatis自定义handler_bc_03

同理,set方法的实现也类似

@Override
     public void setNonNullParameter(PreparedStatement ps, int i, Boolean parameter, JdbcType jdbcType)
             throws SQLException {
             if(parameter) {
                 ps.setInt(i, 1); 
             }else {
                 ps.setInt(i, 0); 
             }
     }

写完转换器之后,需要在config.xml中配置一下,让mybatis能认识这个转换器

<typeHandlers>
         <typeHandler handler="org.lanqiao.converter.BooleanAndIntConverter" javaType="Boolean" jdbcType="INTEGER" />
     </typeHandlers>

在mapper文件中写sql

<!-- 查询:使用了类型转换器
    1如果 类中属性 和表中的字段 类型能够合理识别 (String-varchar2),则可以使用resultType;否则(boolean-number) 使用resultMap
    2如果 类中属性名 和表中的字段名能够合理识别 (stuNo -stuno)则可以使用resultType;否则(id-stuno) 使用resultMap

-->
     <select id="queryStudentByStunoWithConverter" parameterType="int" resultMap="studentResult">
         select * from student2 where stuno = #{stuno}
     </select>

     <resultMap type="student" id="studentResult">                
         <!-- 分为主键id 和非主键 result-->
             <id property="stuNo"  column="stuno"  />
             <result property="stuName"  column="stuname" />
             <result property="stuAge"  column="stuage" />
             <result property="graName"  column="graname" />
             <result property="stuSex" column="stusex" javaType="boolean" jdbcType="INTEGER"/>
     </resultMap>

两种情况下使用resultMap:1.类中的java类型和数据库类型不一致(boolean,number)  2.类中的属性名和表中字段不一致(id,stuno)。

mybatais javaType 自定义 mybatis自定义handler_TypeHandler_04

能够查询出该1编号的BOOLEAN类型

mybatais javaType 自定义 mybatis自定义handler_类型转换器_05

带转换器的add操作,即需要用到setNonNullParameter方法

<!-- 带转换器的增加 -->
     <insert id="addStudentWithConverter" parameterType="student" >
         insert into student2(stuno,stuname,stuage,graname,stusex) values(#{stuNo},#{stuName},#{stuAge},#{graName} ,#{stuSex,javaType=boolean,jdbcType=INTEGER} ) 
     </insert>

测试之后,也能实现添加操作