Mybatis自定义TypeHandler解决特殊类型转换问题详解
我们知道,Java和MySQL中的数据类型是不同的,Java中除了基本数据类型,还有对象。 有时候使用MySQL存储数据,或者从MySQL中读取数据时,会有一些特殊需求 weary ,比如:
这也太难了叭! 解决办法有两种:
关于第一种方法这里不予赘述,不够Smart。这里主要讲述如何自定义Handler,来解决Java数据->MySQL数据的特殊类型转换问题grinning 这种Handler不仅方便了我们的数据库操作,还有利于代码的复用。 这里以Integer[]数组的存储为形如,1,2,3,的varchar字符串为例。 问题示例 我们定义一个role类,与数据库的role表对应: public class Role { private Integer id; private String name; private Integer[] accessIds; private Date createTime; // ... ignore get and set methods } 注意到里面有一个accessIds字段,它的类型是Integer[] 数据库设计: DROP TABLE IF EXISTS `role`; CREATE TABLE `role` ( `id` int(11) NOT NULL AUTO_INCREMENT,`name` varchar(255) NOT NULL,`access_ids` varchar(255) DEFAULT NULL,`create_time` datetime NOT NULL,PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8; -- ---------------------------- -- Records of role -- ---------------------------- INSERT INTO `role` VALUES ('1','测试角色',','2019-11-14 13:43:14'); 自定义Handler类 通过继承BaseTypeHandler类,重写其方法,定义一个Integer[]与数据库varchar类型自动转换的Handler类: /** * Java Int数组与MySQL String转换器 * 比如[1,3] --> "," */ public class StringToIntArrayHandler extends BaseTypeHandler<Integer[]> { private static final String splitCharset = ","; @Override public void setNonNullParameter(PreparedStatement ps,int i,Integer[] objects,JdbcType jdbcType) throws SQLException { String str = arrayToString(objects); ps.setString(i,str); } @Override public Integer[] getNullableResult(ResultSet rs,String columnName) throws SQLException { String str = rs.getString(columnName); return stringToArray(str); } @Override public Integer[] getNullableResult(ResultSet rs,int columnIndex) throws SQLException { String str = rs.getString(columnIndex); return stringToArray(str); } @Override public Integer[] getNullableResult(CallableStatement cs,int columnIndex) throws SQLException { String str = cs.getString(columnIndex); return stringToArray(str); } // --- private methods --- /** * Integer数组转String * 注:使用提前设定好的分隔符分割数组的每一项 */ private static String arrayToString(Integer[] array) { StringBuilder res = new StringBuilder(); if (array != null && array.length > 0) { for (Object o : array) { res.append(splitCharset).append(o.toString()); } res.append(splitCharset); } return res.length() > 0 ? res.toString() : null; } /** * 从String转Integer数组 * 注:String是用分隔符分割的,使用String.split方法可以分解为数组 */ private static Integer[] stringToArray(String str) { List<Integer> list = new ArrayList<>(); if (str != null) { String[] array = str.split(splitCharset); if (array.length > 0) { for (String o : array) { if (o != null && o.length() > 0) { list.add(Integer.parseInt(o)); } } } } return list.toArray(new Integer[0]); } } 这个类的具体作用是什么呢?
下面我们演示一下具体的使用smile 在Mybatis中应用自定义的Handler Mybatis存放SQL语句的XML文件: <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" > <mapper namespace="com.example.model.dao.RoleDAO"> <resultMap id="roleMap" type="com.example.model.bean.Role"> <id property="id" column="id"/> <result property="name" column="name"/> <result property="accessIds" column="access_ids" typeHandler="ccom.example.model.dao.handler.StringToIntArrayHandler"/> <result property="createTime" column="create_time"/> </resultMap> <select id="findById" parameterType="map" resultMap="roleMap"> SELECT id,name,access_ids,create_time FROM role WHERE id = #{id} </select> <insert id="insert" parameterType="com.example.model.bean.Role"> <selectKey resultType="java.lang.Integer" order="AFTER" keyProperty="id"> SELECT LAST_INSERT_ID() </selectKey> INSERT INTO role (name,create_time,access_ids) VALUES (#{name},#{createTime},#{accessIds,jdbcType=VARCHAR,typeHandler=com.example.model.dao.handler.StringToIntArrayHandler}) </insert> </mapper> 以上XML中演示了select和insert两种情况时,如何应用typeHandler。 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |