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

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

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

Mybatis自定义类型转换器(TypeHandler)

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

Mybatis自定义类型转换器(TypeHandler)

其中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拿值,也可以通过存储过程拿,方法都一样。

Mybatis自定义类型转换器(TypeHandler)

同理,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)。

Mybatis自定义类型转换器(TypeHandler)

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

Mybatis自定义类型转换器(TypeHandler)

带转换器的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>

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

相关文章: