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

我们知道,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 {

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 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文件​​:

typeHandler="ccom.example.model.dao.handler.StringToIntArrayHandler"/>

SELECT id, name, access_ids, create_time

FROM role

WHERE id = #{id}

SELECT LAST_INSERT_ID()

INSERT INTO role

(name, create_time, access_ids)

VALUES

(#{name}, #{createTime}

, #{accessIds, jdbcType=VARCHAR, typeHandler=com.example.model.dao.handler.StringToIntArrayHandler})

以上XML中演示了select和insert两种情况时,如何应用typeHandler。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值