Mybatis?TypeHandler接口及繼承關(guān)系示例解析
開篇
JDBC類型與Java類型并不是完全一一對應(yīng)的。所以在PreparedStatement綁定參數(shù)的時候需要把Java類型轉(zhuǎn)為JDBC類型。JDBC類型的枚舉值在JdbcType枚舉值中存儲。
MyBatis中提供了一個接口專用于JDBC類型與Java類型的轉(zhuǎn)換。它就是我們今天的主題:TypeHandler(類型轉(zhuǎn)換器)
TypeHandler接口
TypeHandler是用于JDBC類型與Java類型的轉(zhuǎn)換
我們先來看一下這個接口的定義,它都規(guī)范了哪些行為。再來說這些方法的實(shí)現(xiàn)類和具體作用。
public interface TypeHandler<T> {
void setParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException;
T getResult(ResultSet rs, String columnName) throws SQLException;
T getResult(ResultSet rs, int columnIndex) throws SQLException;
T getResult(CallableStatement cs, int columnIndex) throws SQLException;
}
該接口中共有4個方法。這四個方法只定義了兩種行為,一是setParameter(設(shè)置參數(shù))二是操作結(jié)果集獲取指定對象
| 方法名 | 描述 |
|---|---|
| setParameter | 使用PreparedStatement執(zhí)行SQL是,設(shè)置帶?的參數(shù)。 |
| getResult | 獲取結(jié)果集中指定列名對應(yīng)的值,或者獲取結(jié)果集中指定索引對應(yīng)的值 |
實(shí)際setParameter方法的底層實(shí)現(xiàn)就是JDBC操作
PreparedStatement ps = connection.prepareStatement(); ps.setString(1,"value");
getResult方法的底層同樣也是JDBC操作
// rs是結(jié)果集對象ResultSet rs.getInt(colName); rs.getInt(i)
TypeHandler繼承體系
在MyBatis中JDBC類型被定義在枚舉類JdbcType中。共有41中JDBC類型。每個JDBC類型都對應(yīng)一個XxxTypeHandler類。每個類都是解決特定的JDBC類型轉(zhuǎn)換
TypeHandler接口值規(guī)定了行為,每種JDBC類型,都提供了對應(yīng)的實(shí)現(xiàn)類來完場JDBC到Java類型的轉(zhuǎn)換。

每個TypeHandler的實(shí)現(xiàn)都大同小異,而為了避免空指針的問題,TypeHandler還有一個抽象子類,對這4個方法做了模板處理。
- 設(shè)置參數(shù):如果值為null則直接跳過,否則調(diào)用具體實(shí)現(xiàn)類設(shè)置參數(shù)
- 從結(jié)果集中獲取數(shù)據(jù):調(diào)用具體實(shí)現(xiàn)類的
getNullableResult方法
BaseTypeHandler的代碼比較簡單易懂,這列舉重要實(shí)現(xiàn):
public void setParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException {
if (parameter == null) {
if (jdbcType == null) {拋異常}
try {
ps.setNull(i, jdbcType.TYPE_CODE);
} catch (SQLException e) { 拋異常 }
} else {
try {
// 如果參數(shù)不為Null才調(diào)用子類方法
setNonNullParameter(ps, i, parameter, jdbcType);
} catch (Exception e) { 拋異常 }
}
}
我們先來剖析下常用的類型。IntegerTypeHandler
IntegerTypeHandler
public class IntegerTypeHandler extends BaseTypeHandler<Integer> {
@Override
public void setNonNullParameter(PreparedStatement ps, int i, Integer parameter, JdbcType jdbcType) throws SQLException {
ps.setInt(i, parameter);
}
public Integer getNullableResult(ResultSet rs, String columnName) throws SQLException {
int result = rs.getInt(columnName);
return result == 0 && rs.wasNull() ? null : result;
}
// 省略另外兩個方法
}
可以看到IntegerTypeHandler的底層就是調(diào)用了JDBC對象PreparedStatement的方法來設(shè)置參數(shù)與獲取結(jié)果集中的記錄。這個實(shí)現(xiàn)是最簡單的一個,就不展開介紹了。
下面我們再來看一下日期類型的映射
DateTypeHandler
public class DateTypeHandler extends BaseTypeHandler<Date> {
@Override
public void setNonNullParameter(PreparedStatement ps, int i, Date parameter, JdbcType jdbcType) throws SQLException {
ps.setTimestamp(i, new Timestamp(parameter.getTime()));
}
@Override
public Date getNullableResult(ResultSet rs, String columnName) throws SQLException {
Timestamp sqlTimestamp = rs.getTimestamp(columnName);
if (sqlTimestamp != null) {
return new Date(sqlTimestamp.getTime());
}
return null;
}
}
- 設(shè)置參數(shù):我們使用Java編程,使用的日期常用的類是
java.util.Date,但是在JDBC操作數(shù)據(jù)庫的時候,往往需要的不是Date類型,而是SQL標(biāo)準(zhǔn)定義的Timestamp類型,DateTypeHandler會幫助我們把Date轉(zhuǎn)化為Timestamp - 獲取結(jié)果,從JDBC中獲取的記錄如果是Timestamp類型,則DateTypeHandler幫我們轉(zhuǎn)為
java.util.Date對象
注:可能大家在編程JDBC的時候,執(zhí)行如下語句select * from user where birthday > ?,無論數(shù)據(jù)庫中birthday字段是Date/DateTime/TimeStamp類型,我們都會把會把?的值設(shè)置為Date類型,實(shí)際上這是不符合JDBC標(biāo)準(zhǔn)的。
其他TypeHandler的實(shí)現(xiàn),就不再看了。邏輯都是一樣的。
TypeHandlerRegistry
通過上一小節(jié)了解到了每一種JDBC類型,都提供了TypeHandler的實(shí)現(xiàn)。那么mybatis什么時候才會用到這些實(shí)現(xiàn)類呢?或者說當(dāng)需要進(jìn)行類型轉(zhuǎn)換的時候,MyBatis是如何使用這些實(shí)現(xiàn)類的。
在MyBatis初始化的時候,會加載TypeHandlerRegistry這個類,這個類有在構(gòu)造方法里會調(diào)用一系列的register方法把TypeHandler所有實(shí)現(xiàn)類都注冊到TypeHandlerRegistry中。下面我們先來看一下TypeHandlerRegistry中的重要屬性
public final class TypeHandlerRegistry {
private final Map<JdbcType, TypeHandler<?>> jdbcTypeHandlerMap = new EnumMap<>(JdbcType.class);
private final Map<Type, Map<JdbcType, TypeHandler<?>>> typeHandlerMap = new ConcurrentHashMap<>();
private final TypeHandler<Object> unknownTypeHandler;
private final Map<Class<?>, TypeHandler<?>> allTypeHandlersMap = new HashMap<>();
}
- jdbcTypeHandlerMap:key是Jdbc類型,value是Jdbc類型對應(yīng)的處理器
- typeHandlerMap:key是Java類型,value是對應(yīng)的處理器集合(它也是要給Map)。因?yàn)橐粋€Java類型可能被多個處理器解析。比如String類型可能被解析為char varchar類型等。比如Java的
java.util.Date類型可以被解析為Date Time TimeStamp類型。存在一對多的關(guān)系 - unknownTypeHandler:位置的TypeHandler,一般為空
- allTypeHandlersMap:所有類型處理器,key是TypeHandler實(shí)現(xiàn)類的Class對象,value是具體的TypeHandler
TypeHandlerRegistry#register方法
上面介紹了TypeHandlerRegistry中幾個重要的屬性。在mybatis啟動時,就會把所有的TypeHandler都注冊到如上的4個數(shù)據(jù)結(jié)構(gòu)中。具體的實(shí)現(xiàn)在TypeHandlerRegistry的構(gòu)造方法中,調(diào)用register方法進(jìn)行注冊TypeHandler
public TypeHandlerRegistry(Configuration configuration) {
register(Boolean.class, new BooleanTypeHandler());
register(boolean.class, new BooleanTypeHandler());
register(JdbcType.BOOLEAN, new BooleanTypeHandler());
register(JdbcType.BIT, new BooleanTypeHandler());
register(Byte.class, new ByteTypeHandler());
register(byte.class, new ByteTypeHandler());
register(JdbcType.TINYINT, new ByteTypeHandler());
register(Short.class, new ShortTypeHandler());
register(short.class, new ShortTypeHandler());
register(JdbcType.SMALLINT, new ShortTypeHandler());
// ...省略其他register
}
這些register最終將如上的3個Map數(shù)據(jù)結(jié)構(gòu)填充。通過名字也能看出來,注冊Class/JdbcType與TypeHandler的映射關(guān)系。而這些register分為兩大類,一類是注冊JDBC-TypeHandler映射關(guān)系,一類是注冊Java-TypeHandler的映射關(guān)系。我們先來看第一種
public void register(JdbcType jdbcType, TypeHandler<?> handler) {
jdbcTypeHandlerMap.put(jdbcType, handler);
}
它的實(shí)現(xiàn)很簡單,就是直接把jdbcType作為key,TypeHandler作為value,存入jdbcTypeHandlerMap即可。接下來我們再來看第二種:注冊Java-TypeHandler的映射關(guān)系
private <T> void register(Type javaType, TypeHandler<? extends T> typeHandler) {
MappedJdbcTypes mappedJdbcTypes = typeHandler.getClass().getAnnotation(MappedJdbcTypes.class);
if (mappedJdbcTypes != null) {
for (JdbcType handledJdbcType : mappedJdbcTypes.value()) {
register(javaType, handledJdbcType, typeHandler);
}
if (mappedJdbcTypes.includeNullJdbcType()) {
register(javaType, null, typeHandler);
}
} else {
register(javaType, null, typeHandler);
}
}
我們來分析下 注冊步驟:
- 獲取MappedJdbcTypes注解,對自定義的TypeHandler進(jìn)行注冊。(自定義TypeHandler時需要該注解指定自定義的TypeHandler都解析哪些JDBC類型)
- 如果TypeHandler沒有注解信息(也就是沒有自定義TypeHandler)則直接調(diào)用重載register方法注冊Java-TypeHandler的映射關(guān)系。
下面就來看一下這個重載register方法
private void register(Type javaType, JdbcType jdbcType, TypeHandler<?> handler) {
if (javaType != null) {
Map<JdbcType, TypeHandler<?>> map = typeHandlerMap.get(javaType);
if (map == null || map == NULL_TYPE_HANDLER_MAP) {
map = new HashMap<>();
}
map.put(jdbcType, handler);
typeHandlerMap.put(javaType, map);
}
allTypeHandlersMap.put(handler.getClass(), handler);
}
代碼中它主要做了兩件事:
- 注冊Java-TypeHandler的映射關(guān)系
- 把TypeHandler的Class和類的映射關(guān)系存入allTypeHandlersMap
至此TypeHandlerRegistry中就有了最重要的兩種映射關(guān)系
- JDBC類型 與 TypeHandler 的映射關(guān)系(通過Jdbc類型尋找合適的TypeHandler類型,通常用于結(jié)果集的解析過程)
- Java類型 與 TypeHandler 的映射關(guān)系(通過Java類型尋找合適的TypeHandler類型,通常用于為PreparedStatement對象設(shè)置參數(shù))
TypeHandlerRegistry也提供了一系列的get*方法,可以根據(jù)指定的信息返回需要的TypeHandler類
- getMappingTypeHandler(Class>):根據(jù)TypeHandler類型獲取對應(yīng)的TypeHandler對象
- getTypeHandler(JdbcType):根據(jù)JdbcType類型獲取TypeHandler對象
- getTypeHandler(Class):根據(jù)Class類型獲取TypeHandler對象
- getJdbcHandlerMap(Type):根據(jù)Java類型獲取TypeHandler對象
總結(jié)
TypeHandler是類型處理器,它用來解析Java類型與Jdbc類型之間的相互轉(zhuǎn)換。
TypeHandlerRegistry是一個注冊器,其中注冊了JDBC與TypeHandler的映射關(guān)系、Java類型與TypeHandler的映射關(guān)系。
那么由此我們可以想象到,在mybatis執(zhí)行SQL的過程中,一定會在某處調(diào)用TypeHandlerRegistry并通過參數(shù)的Java類型獲取對應(yīng)的TypeHandler對象為PreparedStatement設(shè)置參數(shù)。
也一定會在解析結(jié)果集的過程中,調(diào)用TypeHandlerRegistry并通過結(jié)果集數(shù)據(jù)的Jdbc類型獲取對應(yīng)的TypeHandler對象為來解析結(jié)果集中的記錄
以上就是Mybatis TypeHandler接口及繼承關(guān)系示例解析的詳細(xì)內(nèi)容,更多關(guān)于Mybatis TypeHandler接口繼承的資料請關(guān)注腳本之家其它相關(guān)文章!
- Mybatis的TypeHandler實(shí)現(xiàn)數(shù)據(jù)加解密詳解
- Mybatis基于TypeHandler實(shí)現(xiàn)敏感數(shù)據(jù)加密
- MyBatis中的自定義TypeHandler詳解
- MyBatis?typeHandler接口的定義和使用
- Mybatis之類型處理器TypeHandler的作用與自定義方式
- mybatis之BaseTypeHandler用法解讀
- mybatis-plus之自動映射字段(typeHandler)的注意點(diǎn)及說明
- Mybatis的類型轉(zhuǎn)換接口TypeHandler
- Mybatis中TypeHandler使用小結(jié)
相關(guān)文章
Java編程中二維數(shù)組的初始化和基本操作實(shí)例
這篇文章主要介紹了Java編程中二維數(shù)組的初始化和基本操作實(shí)例,是Java入門學(xué)習(xí)中的基礎(chǔ)知識,需要的朋友可以參考下2015-10-10
InputStreamReader 和FileReader的區(qū)別及InputStream和Reader的區(qū)別
這篇文章主要介紹了InputStreamReader 和FileReader的區(qū)別及InputStream和Reader的區(qū)別的相關(guān)資料,需要的朋友可以參考下2015-12-12
SpringBoot創(chuàng)建WebService方法詳解
這篇文章主要介紹了SpringBoot如何創(chuàng)建WebService,文中有詳細(xì)的實(shí)現(xiàn)步驟以及示例代碼,對學(xué)習(xí)或工作有一定的幫助,需要的朋友跟著小編一起來學(xué)習(xí)吧2023-05-05
解析spring事務(wù)管理@Transactional為什么要添加rollbackFor=Exception.class
這篇文章主要介紹了spring事務(wù)管理@Transactional為什么要添加rollbackFor=Exception.class,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-11-11
如何將Java對象轉(zhuǎn)換為JSON實(shí)例詳解
有時候需要將對象轉(zhuǎn)換為JSON格式,所以這篇文章主要給大家介紹了關(guān)于如何將Java對象轉(zhuǎn)換為JSON的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-08-08

