Mybatis自定义TypeHandler解决特殊类型转换问题详解

作者:空夜 时间:2023-04-14 09:50:51 

我们知道,Java和MySQL中的数据类型是不同的,Java中除了基本数据类型,还有对象。

有时候使用MySQL存储数据,或者从MySQL中读取数据时,会有一些特殊需求 weary ,比如:

  1. 将Integer数组直接存入MySQL,保存为BLOB形式,读取出来时又是正常的Integer数组

  2. 将Integer数组转换为String,然后存入MySQL,使用varchar类型,读取出来时又是正常的Integer数组

这也太难了叭!

解决办法有两种:

  1. Basic Method:Java在存入数据之前,或读取数据之后,做手动类型转换

  2. Clever Method:定义TypeHandler,并在Mybatis对应位置指明

关于第一种方法这里不予赘述,不够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', '测试角色', ',1,2,', '2019-11-14 13:43:14');

自定义Handler类

通过继承BaseTypeHandler类,重写其方法,定义一个Integer[]与数据库varchar类型自动转换的Handler类:


/**
* Java Int数组与MySQL String转换器
* 比如[1,2,3] --> ",1,2,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]);
 }
}

这个类的具体作用是什么呢?

  1. 当Java中类型是Integer[]时,使用这个Handler类,将Integer[]转换为以,号分割的字符串,然后存入数据库

  2. 当从数据库读取以,分割值的字符串时,可以通过这个Handler,自动将字符串转换为Integer[]数组

下面我们演示一下具体的使用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。

来源:https://segmentfault.com/a/1190000021004500

标签:Mybatis,TypeHandler,类型转换
0
投稿

猜你喜欢

  • C#中英文混合字符串截取函数

    2023-01-19 06:02:55
  • C# Winform按钮中图片实现左图右字的效果实例

    2022-05-01 12:51:41
  • Spring Boot2.X国际化文件编写配置

    2023-02-24 11:34:24
  • Java中闭包简单代码示例

    2023-11-08 23:09:48
  • Jenkins 关闭和重启详细介绍及实现

    2022-09-19 00:57:12
  • 5步学会使用VideoView播放视频

    2023-09-12 05:51:07
  • Android快速实现发送邮件实例

    2022-06-10 03:39:48
  • springboot编程式事务TransactionTemplate的使用说明

    2022-03-01 15:19:37
  • jbuilder2006连接sqlserver2000的方法

    2022-08-21 14:20:09
  • Java并发编程之浅谈ReentrantLock

    2022-08-25 10:46:02
  • java模拟http的Get/Post请求,并设置ip与port代理的方法

    2021-11-25 11:30:57
  • JAVA使用POI(XSSFWORKBOOK)读取EXCEL文件过程解析

    2023-03-01 16:36:39
  • Android开发之ViewPager实现滑动切换页面

    2023-03-10 05:32:43
  • Android 三级NestedScroll嵌套滚动实践

    2022-11-12 07:45:21
  • Android PowerManagerService省电模式策略控制

    2023-11-25 02:46:53
  • java实现删除某条信息并刷新当前页操作

    2022-06-26 07:12:12
  • Android RecyclerView多类型布局卡片解决方案

    2022-06-18 17:39:09
  • Java 8 Stream操作类型及peek示例解析

    2021-07-17 20:42:08
  • C#中WebBroeser控件用法实例教程

    2021-09-14 18:11:05
  • 基于DateTime.ParseExact方法的使用详解

    2021-11-15 11:33:39
  • asp之家 软件编程 m.aspxhome.com