- 資訊首頁(yè) > 開(kāi)發(fā)技術(shù) >
- Mybatis如何自定義TypeHandler解決特殊類(lèi)型轉換問(wèn)題
這篇文章給大家分享的是有關(guān)Mybatis如何自定義TypeHandler解決特殊類(lèi)型轉換問(wèn)題的內容。小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過(guò)來(lái)看看吧。
我們知道,Java和中的數據類(lèi)型是不同的,Java中除了基本數據類(lèi)型,還有對象。
有時(shí)候使用MySQL存儲數據,或者從MySQL中讀取數據時(shí),會(huì )有一些特殊需求 weary ,比如:
將Integer數組直接存入MySQL,保存為BLOB形式,讀取出來(lái)時(shí)又是正常的Integer數組 將Integer數組轉換為String,然后存入MySQL,使用varchar類(lèi)型,讀取出來(lái)時(shí)又是正常的Integer數組
這也太難了叭!
解決辦法有兩種:
Basic Method:Java在存入數據之前,或讀取數據之后,做手動(dòng)類(lèi)型轉換 Clever Method:定義TypeHandler,并在Mybatis對應位置指明
關(guān)于第一種方法這里不予贅述,不夠Smart。這里主要講述如何自定義Handler,來(lái)解決Java數據->MySQL數據的特殊類(lèi)型轉換問(wèn)題grinning
這種Handler不僅方便了我們的數據庫操作,還有利于代碼的復用。
這里以Integer[]數組的存儲為形如,1,2,3,的varchar字符串為例。
問(wèn)題示例
我們定義一個(gè)role類(lèi),與數據庫的role表對應:
public class Role { private Integer id; private String name; private Integer[] accessIds; private Date createTime; // ... ignore get and set methods}
注意到里面有一個(gè)accessIds字段,它的類(lèi)型是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類(lèi)
通過(guò)繼承BaseTypeHandler類(lèi),重寫(xiě)其方法,定義一個(gè)Integer[]與數據庫varchar類(lèi)型自動(dòng)轉換的Handler類(lèi):
/** * 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]); }}
這個(gè)類(lèi)的具體作用是什么呢?
當Java中類(lèi)型是Integer[]時(shí),使用這個(gè)Handler類(lèi),將Integer[]轉換為以,號分割的字符串,然后存入數據庫 當從數據庫讀取以,分割值的字符串時(shí),可以通過(guò)這個(gè)Handler,自動(dòng)將字符串轉換為Integer[]數組
下面我們演示一下具體的使用smile
在Mybatis中應用自定義的Handler
Mybatis存放SQL語(yǔ)句的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>
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng )、來(lái)自互聯(lián)網(wǎng)轉載和分享為主,文章觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權請聯(lián)系QQ:712375056 進(jìn)行舉報,并提供相關(guān)證據,一經(jīng)查實(shí),將立刻刪除涉嫌侵權內容。
Copyright ? 2009-2021 56dr.com. All Rights Reserved. 特網(wǎng)科技 特網(wǎng)云 版權所有 珠海市特網(wǎng)科技有限公司 粵ICP備16109289號
域名注冊服務(wù)機構:阿里云計算有限公司(萬(wàn)網(wǎng)) 域名服務(wù)機構:煙臺帝思普網(wǎng)絡(luò )科技有限公司(DNSPod) CDN服務(wù):阿里云計算有限公司 中國互聯(lián)網(wǎng)舉報中心 增值電信業(yè)務(wù)經(jīng)營(yíng)許可證B2
建議您使用Chrome、Firefox、Edge、IE10及以上版本和360等主流瀏覽器瀏覽本網(wǎng)站