Mybatis自定義TypeHandler解決特殊類(lèi)型轉(zhuǎn)換問(wèn)題詳解
我們知道,Java和MySQL中的數(shù)據(jù)類(lèi)型是不同的,Java中除了基本數(shù)據(jù)類(lèi)型,還有對(duì)象。
有時(shí)候使用MySQL存儲(chǔ)數(shù)據(jù),或者從MySQL中讀取數(shù)據(jù)時(shí),會(huì)有一些特殊需求 weary ,比如:
- 將Integer數(shù)組直接存入MySQL,保存為BLOB形式,讀取出來(lái)時(shí)又是正常的Integer數(shù)組
- 將Integer數(shù)組轉(zhuǎn)換為String,然后存入MySQL,使用varchar類(lèi)型,讀取出來(lái)時(shí)又是正常的Integer數(shù)組
這也太難了叭!
解決辦法有兩種:
- Basic Method:Java在存入數(shù)據(jù)之前,或讀取數(shù)據(jù)之后,做手動(dòng)類(lèi)型轉(zhuǎn)換
- Clever Method:定義TypeHandler,并在Mybatis對(duì)應(yīng)位置指明
關(guān)于第一種方法這里不予贅述,不夠Smart。這里主要講述如何自定義Handler,來(lái)解決Java數(shù)據(jù)->MySQL數(shù)據(jù)的特殊類(lèi)型轉(zhuǎn)換問(wèn)題grinning
這種Handler不僅方便了我們的數(shù)據(jù)庫(kù)操作,還有利于代碼的復(fù)用。
這里以Integer[]數(shù)組的存儲(chǔ)為形如,1,2,3,的varchar字符串為例。
問(wèn)題示例
我們定義一個(gè)role類(lèi),與數(shù)據(jù)庫(kù)的role表對(duì)應(yīng):
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[]
數(shù)據(jù)庫(kù)設(shè)計(jì):
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', '測(cè)試角色', ',1,2,', '2019-11-14 13:43:14');
自定義Handler類(lèi)
通過(guò)繼承BaseTypeHandler類(lèi),重寫(xiě)其方法,定義一個(gè)Integer[]與數(shù)據(jù)庫(kù)varchar類(lèi)型自動(dòng)轉(zhuǎn)換的Handler類(lèi):
/** * Java Int數(shù)組與MySQL String轉(zhuǎn)換器 * 比如[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數(shù)組轉(zhuǎn)String * 注:使用提前設(shè)定好的分隔符分割數(shù)組的每一項(xiàng) */ 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轉(zhuǎn)Integer數(shù)組 * 注:String是用分隔符分割的,使用String.split方法可以分解為數(shù)組 */ 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)的具體作用是什么呢?
- 當(dāng)Java中類(lèi)型是Integer[]時(shí),使用這個(gè)Handler類(lèi),將Integer[]轉(zhuǎn)換為以,號(hào)分割的字符串,然后存入數(shù)據(jù)庫(kù)
- 當(dāng)從數(shù)據(jù)庫(kù)讀取以,分割值的字符串時(shí),可以通過(guò)這個(gè)Handler,自動(dòng)將字符串轉(zhuǎn)換為Integer[]數(shù)組
下面我們演示一下具體的使用smile
在Mybatis中應(yīng)用自定義的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>
以上XML中演示了select和insert兩種情況時(shí),如何應(yīng)用typeHandler。
- mybatis之BaseTypeHandler用法解讀
- mybatis-plus之自動(dòng)映射字段(typeHandler)的注意點(diǎn)及說(shuō)明
- MyBatisPlus?TypeHandler自定義字段類(lèi)型轉(zhuǎn)換Handler
- mybatis TypeHandler注入spring的依賴(lài)方式
- MybatisPlus如何自定義TypeHandler映射JSON類(lèi)型為L(zhǎng)ist
- MyBatis自定義typeHandler的完整實(shí)例
- Mybatis實(shí)現(xiàn)自定義類(lèi)型轉(zhuǎn)換器TypeHandler的方法
- Mybatis實(shí)現(xiàn)自定義的typehandler三步曲
- 微服務(wù)mybatis typehandler使用詳解(就這一篇夠了)
相關(guān)文章
Springboot攔截器如何獲取@RequestBody參數(shù)
這篇文章主要介紹了Springboot攔截器如何獲取@RequestBody參數(shù)的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06Windows下Java環(huán)境配置的超詳細(xì)教程
這篇文章主要給大家介紹了關(guān)于Windows下Java環(huán)境配置的超詳細(xì)教程,文中通過(guò)圖文將配置的過(guò)程介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2023-01-01Java 排序算法整合(冒泡,快速,希爾,拓?fù)?,歸并)
這篇文章主要介紹了Java 排序算法整合(冒泡,快速,希爾,拓?fù)?,歸并),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09詳解Spring Boot 項(xiàng)目中的 parent
這篇文章主要介紹了Spring Boot中parent作用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04如何通過(guò)idea實(shí)現(xiàn)springboot集成mybatis
這篇文章主要介紹了如何通過(guò)idea實(shí)現(xiàn)springboot集成mybatis,使用springboot 集成 mybatis后,通過(guò)http請(qǐng)求接口,使得通過(guò)http請(qǐng)求可以直接操作數(shù)據(jù)庫(kù),本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2023-09-09Spring Boot中Reactor模型的基本概念和最佳實(shí)踐
Reactor模型是一種基于事件驅(qū)動(dòng)和非阻塞IO的編程模型,用于處理并發(fā)和異步操作,本文將介紹Spring Boot中使用Reactor模型的基本概念和最佳實(shí)踐,幫助讀者更好地理解如何利用這一強(qiáng)大的工具來(lái)構(gòu)建現(xiàn)代化的Java應(yīng)用程序,感興趣的朋友跟隨小編一起看看吧2024-05-05MyBatis實(shí)現(xiàn)多表聯(lián)查的詳細(xì)代碼
這篇文章主要介紹了MyBatis如何實(shí)現(xiàn)多表聯(lián)查,通過(guò)實(shí)例代碼給大家介紹使用映射配置文件實(shí)現(xiàn)多表聯(lián)查,使用注解的方式實(shí)現(xiàn)多表聯(lián)查,需要的朋友可以參考下2022-08-08Java動(dòng)態(tài)代理實(shí)現(xiàn)方法小結(jié)
這篇文章主要介紹了Java動(dòng)態(tài)代理實(shí)現(xiàn)方法,結(jié)合實(shí)例形式總結(jié)分析了java基于JDK、CGLIB及CGLIB實(shí)現(xiàn)動(dòng)態(tài)代理的相關(guān)操作技巧,需要的朋友可以參考下2019-02-02