Mybatis自定义TypeHandler解决特殊类型转换问题详解
作者:空夜 发布时间:2023-04-14 09:50:51
我们知道,Java和MySQL中的数据类型是不同的,Java中除了基本数据类型,还有对象。
有时候使用MySQL存储数据,或者从MySQL中读取数据时,会有一些特殊需求 weary ,比如:
将Integer数组直接存入MySQL,保存为BLOB形式,读取出来时又是正常的Integer数组
将Integer数组转换为String,然后存入MySQL,使用varchar类型,读取出来时又是正常的Integer数组
这也太难了叭!
解决办法有两种:
Basic Method:Java在存入数据之前,或读取数据之后,做手动类型转换
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]);
}
}
这个类的具体作用是什么呢?
当Java中类型是Integer[]时,使用这个Handler类,将Integer[]转换为以,号分割的字符串,然后存入数据库
当从数据库读取以,分割值的字符串时,可以通过这个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
猜你喜欢
- Maven修改打包文件名称对Maven打出的jar包名称不满意:想通过修改配置给jar包改名,查询找到了方法:pom.xml的<bui
- 最近同事问我有没有有关于技术的电子书,我打开电脑上的小书库,但是邮件发给他太大了,公司又禁止用文件夹共享,于是花半天时间写了个小的文件上传程
- 实例如下:package com.huad.luck;import java.util.ArrayList;import java.util
- Spring AOP后置通知修改响应httpstatus1.定义Aspect/** * 响应体切面 * 后置通知修改httpstatus *
- 前言今天在逛某知名论坛的时候,看到一篇"请不要使用包装类型,避免造成性能损失"的文章。一下子就吸引了我的注意。大意就是,
- 1.动态绑定机制java的动态绑定机制非常重要实例A我们来看一个实例:阅读上面的代码,请说明下面的程序将输出什么结果:程序将会输出40和30
- 本文实例讲述了Java基于链表实现栈的方法。分享给大家供大家参考,具体如下:在上几小节中我们实现了基本的链表结构,并在上一节的底部给出了有关
- 过去的每一year,涌现出越来越多的Java框架。就像JavaScript,每个人都认为他们知道一个好的框架的功能应该是怎么样的。连我的老祖
- java Mybatis存进时间戳封装了一个实体类,里面有个字段 Integer createTime。要利用这个实体类将一个时间戳存进数据
- 写在开头java.util.Collection 作为Java开发最常用的接口之一,我们经常使用,今天我带大家一起研究一下Collectio
- 一、方法的定义1.方法体中最后返回值可以使用return, 如果使用了return, 那么方法体的返回值类型一定要指定2.如果方法体重没有r
- 一、前言今天实现一个时钟工具,其实在之前已经完成了一个简单的时钟工具:【Unity3D应用案例系列】时钟、钟表小组件开发。那么,今天的这个小
- Java xml出现错误 javax.xml.transform.TransformerException: java.lang.NullP
- 实现功能:模拟简单登录功能,登录成功跳转新页面,登录失败在原登录界面提示登录失败信息开发环境:eclipseTomcat-8.0预备知识:H
- try &
- 接着上一篇进行学习java文件上传下载1。五、断点续传 对于熟用QQ的程序员,QQ的断点续传功能应该是印象很深刻的。因为它很实用也
- InputStreamReader 类1、概述转换流 java.io.InputStreamReader ,是Reader的子类,是从字节流
- 前言前两篇我们详细了解了 findById 和 findAll 以及 findAll 的分页查询,如果说JPA只有上面的两种查询功能,那就太
- 一、概述针对八种基本数据类型定义相应的引用类型—包装类(封装类)。二、作用有了类的特点,就可以调用类中的方法,Java才
- mybatis if test判断BigDecimal遇到的坑<update id="test" paramete