当前位置: 首页 > news >正文

MyBatis 源码解析:TypeHandler 设计与自定义实现


引言

在 MyBatis 中,TypeHandler 是一个非常重要的接口,它的作用是将 Java 类型和数据库类型进行互相转换。当我们执行 SQL 查询或插入操作时,TypeHandler 决定了如何将 Java 对象转换为数据库字段类型,或将数据库字段转换为 Java 对象。MyBatis 内置了多种 TypeHandler 来处理常见的类型转换,但在一些特定场景下,我们可能需要自定义 TypeHandler 来处理特殊的数据类型。在本篇文章中,我们将通过自定义实现一个 TypeHandler,展示它的设计与应用。

摘要

TypeHandler 是 MyBatis 中用于将 Java 类型与数据库字段类型进行转换的接口。本文将通过自定义实现一个 TypeHandler,展示如何将 Java 对象转换为数据库字段,反之亦然,并与 MyBatis 内置的 TypeHandler 机制进行对比,帮助读者掌握 TypeHandler 的原理及其应用场景。

什么是 MyBatis 中的 TypeHandler

TypeHandler 是 MyBatis 的一个接口,它的作用是在将数据存储到数据库时,将 Java 对象转换为数据库能够识别的类型;反之,在从数据库查询数据时,将数据库中的数据转换为 Java 对象。

MyBatis 提供了许多内置的 TypeHandler,例如:

  • IntegerTypeHandler:处理 Java 的 Integer 类型与数据库中的整数类型的映射。
  • StringTypeHandler:处理 Java 的 String 类型与数据库中的字符串类型的映射。

但有时我们需要处理自定义类型,比如将枚举类型映射到数据库的整数值,或者将自定义的对象映射到 JSON 字符串。这时,我们可以通过自定义 TypeHandler 来实现这些特殊的转换逻辑。

MyBatis 中的 TypeHandler 接口

TypeHandler 接口有四个核心方法:

public interface TypeHandler<T> {// 设置参数,将 Java 类型转换为数据库类型void setParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException;// 从数据库结果集中获取结果,将数据库类型转换为 Java 类型T getResult(ResultSet rs, String columnName) throws SQLException;// 从数据库结果集中获取结果(通过列索引),将数据库类型转换为 Java 类型T getResult(ResultSet rs, int columnIndex) throws SQLException;// 从 CallableStatement 中获取结果T getResult(CallableStatement cs, int columnIndex) throws SQLException;
}
  • setParameter():将 Java 类型的参数设置到 PreparedStatement 中。
  • getResult():从 ResultSet 中获取指定列的结果,并将其转换为 Java 对象。

接下来我们将通过自定义一个 TypeHandler 实现,展示如何将枚举类型映射到数据库中的整数类型。

自定义实现 TypeHandler

需求场景

假设我们有一个用户角色枚举 UserRole,它在数据库中存储为整数值(1 表示管理员,2 表示普通用户)。我们需要实现一个自定义的 TypeHandler,将 Java 中的 UserRole 枚举类型与数据库中的整数进行转换。

定义枚举类型

首先,定义一个简单的 UserRole 枚举类型:

/*** 用户角色枚举类*/
public enum UserRole {ADMIN(1),USER(2);private final int code;UserRole(int code) {this.code = code;}public int getCode() {return code;}// 通过代码获取对应的枚举值public static UserRole getByCode(int code) {for (UserRole role : values()) {if (role.getCode() == code) {return role;}}throw new IllegalArgumentException("Unknown code for UserRole: " + code);}
}

实现自定义 TypeHandler

接下来,我们实现 UserRoleTypeHandler,用于将 UserRole 枚举与数据库中的整数值进行转换。

import org.apache.ibatis.type.JdbcType;
import org.apache.ibatis.type.TypeHandler;
import java.sql.*;/*** 自定义的 TypeHandler,用于将 UserRole 枚举类型与数据库中的整数进行转换*/
public class UserRoleTypeHandler implements TypeHandler<UserRole> {@Overridepublic void setParameter(PreparedStatement ps, int i, UserRole parameter, JdbcType jdbcType) throws SQLException {// 将 UserRole 枚举类型转换为数据库中的整数类型if (parameter != null) {ps.setInt(i, parameter.getCode());} else {ps.setNull(i, Types.INTEGER);}}@Overridepublic UserRole getResult(ResultSet rs, String columnName) throws SQLException {// 从结果集中获取整数值,并转换为 UserRole 枚举int code = rs.getInt(columnName);if (rs.wasNull()) {return null;}return UserRole.getByCode(code);}@Overridepublic UserRole getResult(ResultSet rs, int columnIndex) throws SQLException {// 从结果集中通过列索引获取整数值,并转换为 UserRole 枚举int code = rs.getInt(columnIndex);if (rs.wasNull()) {return null;}return UserRole.getByCode(code);}@Overridepublic UserRole getResult(CallableStatement cs, int columnIndex) throws SQLException {// 从存储过程中获取整数值,并转换为 UserRole 枚举int code = cs.getInt(columnIndex);if (cs.wasNull()) {return null;}return UserRole.getByCode(code);}
}

说明

  • setParameter() 方法负责将 UserRole 枚举类型转换为对应的整数,并设置到 PreparedStatement 中。
  • getResult() 方法从 ResultSet 中获取数据库字段的整数值,并将其转换为 UserRole 枚举类型。

注册自定义 TypeHandler

在 MyBatis 中,自定义的 TypeHandler 需要在 mybatis-config.xml 配置文件中进行注册。

<typeHandlers><!-- 注册自定义的 UserRoleTypeHandler --><typeHandler handler="com.example.type.UserRoleTypeHandler" javaType="com.example.enums.UserRole" jdbcType="INTEGER"/>
</typeHandlers>

或者通过注解的方式在 Mapper 接口中指定自定义的 TypeHandler

@Select("SELECT id, name, role FROM users WHERE id = #{id}")
@Results({@Result(column = "role", property = "role", typeHandler = UserRoleTypeHandler.class)
})
User getUserById(int id);

测试自定义 TypeHandler

我们创建一个测试类,验证自定义的 UserRoleTypeHandler 是否能够正常工作。

public class UserRoleTypeHandlerTest {public static void main(String[] args) throws Exception {// 初始化 MyBatis 配置SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(Resources.getResourceAsReader("mybatis-config.xml"));try (SqlSession session = sqlSessionFactory.openSession()) {UserMapper userMapper = session.getMapper(UserMapper.class);// 查询用户并输出角色信息User user = userMapper.getUserById(1);System.out.println("User Role: " + user.getRole()); // 输出角色枚举}}
}

测试结果
当从数据库中查询用户时,UserRoleTypeHandler 会将数据库中的整数字段转换为 UserRole 枚举,并将其设置到 Java 对象中。

类图与流程图

为了更好地理解 TypeHandler 的设计和应用场景,我们提供了类图和流程图。

类图
TypeHandler<T>
+setParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType)
+T getResult(ResultSet rs, String columnName)
+T getResult(ResultSet rs, int columnIndex)
+T getResult(CallableStatement cs, int columnIndex)
UserRoleTypeHandler
+setParameter(PreparedStatement ps, int i, UserRole parameter, JdbcType jdbcType)
+UserRole getResult(ResultSet rs, String columnName)
+UserRole getResult(ResultSet rs, int columnIndex)
+UserRole getResult(CallableStatement cs, int columnIndex)
流程图
setParameter
getResult
getResult
Java对象
数据库类型
Java对象
转换为枚举

MyBatis 中的 TypeHandler 解析

在 MyBatis 中,TypeHandler 是一个用于处理 Java 对象与数据库字段类型之间转换的重要机制。MyBatis 内置了多种 TypeHandler,用于处理常见的 Java 类型和数据库类型的映射,例如 StringTypeHandlerDateTypeHandler 等。

MyBatis 的 TypeHandlerRegistry

MyBatis 通过 TypeHandlerRegistry 来管理所有的 TypeHandler。当执行 SQL 操作时,MyBatis 会根据参数类型和数据库字段类型,自动选择合适的 TypeHandler 进行转换。

public class TypeHandlerRegistry {private final Map<JdbcType, TypeHandler<?>> jdbcTypeHandlerMap = new EnumMap<>(JdbcType.class);public void register(Class<?> javaType, JdbcType jdbcType, TypeHandler<?> handler) {typeHandlerMap.put(javaType, handler);}public <T> TypeHandler<T> getTypeHandler(Class<T> javaType, JdbcType jdbcType) {return typeHandlerMap.get(javaType);}
}

对比分析:手动实现与 MyBatis 的区别

  1. 功能复杂度

    • MyBatis:MyBatis 提供了内置的 TypeHandler,能够处理大部分常见的数据类型映射,并支持自定义扩展。
    • 简化实现:我们的自定义实现展示了如何处理枚举类型的转换,虽然简单但灵活。
  2. 自动注册

    • MyBatis:MyBatis 能够自动根据 Java 类型和数据库类型选择合适的 TypeHandler
    • 简化实现:我们的实现需要手动注册 TypeHandler,适用于自定义类型。
  3. 扩展性

    • MyBatis:MyBatis 支持非常灵活的 TypeHandler 扩展机制,适用于复杂的场景。
    • 简化实现:我们通过展示简单的枚举映射,帮助理解 TypeHandler 的设计和应用。

总结

通过自定义实现一个简单的 TypeHandler,我们展示了如何在 MyBatis 中处理 Java 类型和数据库类型之间的转换。在实际开发中,TypeHandler 是 MyBatis 非常强大的扩展机制,允许开发者在需要时自定义复杂的类型映射。理解这一机制将帮助您在项目中更好地处理复杂的数据库字段和 Java 对象的映射问题。


互动与思考

你是否在项目中遇到过需要自定义类型转换的场景?你认为自定义 TypeHandler 在实际项目中有哪些应用?欢迎在评论区分享你的经验与见解!


如果你觉得这篇文章对你有帮助,请别忘了:

  • 点赞
  • 收藏 📁
  • 关注 👀

让我们一起深入学习 MyBatis 框架,成为更优秀的开发者!


相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 《深度学习》卷积神经网络CNN 原理及其流程解析
  • Ubuntu 22.04.5 LTS 发布下载 - 现代化的企业与开源 Linux
  • 如何在Linux Centos7系统中挂载群晖共享文件夹
  • Kalman算法、扩展卡尔曼滤波(EKF)和无迹卡尔曼滤波(UKF)的比较
  • RuntimeError: Maximum Recursion Depth Exceeded - 递归深度超限的完美解决方案
  • 二叉树堆的建立与排序
  • linux之mysql安装
  • 整数二分算法和浮点数二分算法
  • SpringBootWeb增删改查入门案例
  • ROS 设置dhcp option 6 多个地址格式
  • Qt构建JSON及解析JSON
  • Cesium 绘制可编辑点
  • 新增用户 开发
  • Gin框架入门(2)--异常捕获与日志实现
  • 【系统架构设计师】论文模板:快速写好一篇架构设计师论文
  • golang中接口赋值与方法集
  • Phpstorm怎样批量删除空行?
  • Shadow DOM 内部构造及如何构建独立组件
  • vue-loader 源码解析系列之 selector
  • 初识MongoDB分片
  • 离散点最小(凸)包围边界查找
  • 前端面试题总结
  • 容器化应用: 在阿里云搭建多节点 Openshift 集群
  • 为物联网而生:高性能时间序列数据库HiTSDB商业化首发!
  • 关于Kubernetes Dashboard漏洞CVE-2018-18264的修复公告
  • 国内开源镜像站点
  • # Apache SeaTunnel 究竟是什么?
  • # Java NIO(一)FileChannel
  • #【QT 5 调试软件后,发布相关:软件生成exe文件 + 文件打包】
  • #中的引用型是什么意识_Java中四种引用有什么区别以及应用场景
  • (2)STM32单片机上位机
  • (NO.00004)iOS实现打砖块游戏(十二):伸缩自如,我是如意金箍棒(上)!
  • (八)光盘的挂载与解挂、挂载CentOS镜像、rpm安装软件详细学习笔记
  • (博弈 sg入门)kiki's game -- hdu -- 2147
  • (差分)胡桃爱原石
  • (超简单)使用vuepress搭建自己的博客并部署到github pages上
  • (附源码)springboot高校宿舍交电费系统 毕业设计031552
  • (附源码)小程序 交通违法举报系统 毕业设计 242045
  • (论文阅读23/100)Hierarchical Convolutional Features for Visual Tracking
  • (数位dp) 算法竞赛入门到进阶 书本题集
  • (源码版)2024美国大学生数学建模E题财产保险的可持续模型详解思路+具体代码季节性时序预测SARIMA天气预测建模
  • (转)从零实现3D图像引擎:(8)参数化直线与3D平面函数库
  • .Net 4.0并行库实用性演练
  • .net core Redis 使用有序集合实现延迟队列
  • .net framework profiles /.net framework 配置
  • .NET Framework、.NET Core 、 .NET 5、.NET 6和.NET 7 和.NET8 简介及区别
  • .NET 反射的使用
  • .Net 转战 Android 4.4 日常笔记(4)--按钮事件和国际化
  • .NET/C# 使用反射注册事件
  • .NETCORE 开发登录接口MFA谷歌多因子身份验证
  • @Autowired @Resource @Qualifier的区别
  • @Autowired标签与 @Resource标签 的区别
  • @Bean有哪些属性
  • @selector(..)警告提示
  • [ C++ ] STL---string类的使用指南