news 2026/4/15 8:53:14

MyBatis基础入门《十一》TypeHandler 详解:自定义类型处理器,打通数据库与 Java 的“任督二脉”

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
MyBatis基础入门《十一》TypeHandler 详解:自定义类型处理器,打通数据库与 Java 的“任督二脉”

前情回顾
在 《MyBatis基础入门《十》Spring Boot 整合 MyBatis》 中,我们完成了企业级项目的基础搭建。
但现实业务中,数据库字段和 Java 对象往往不是简单的一一对应

  • MySQL 的JSON字段要映射为Map<String, Object>或自定义对象;
  • 枚举值(如OrderStatus.PAID)需存为数字或字符串;
  • 敏感字段(手机号、身份证)入库前加密,查询后解密。

如何统一、安全、高效地处理这些转换?

答案:使用MyBatis TypeHandler
本文将从原理到实战,手把手教你编写自定义类型处理器。


一、什么是 TypeHandler?

TypeHandler 是 MyBatis 提供的类型转换器,负责:

  • Java Type → JDBC Type(设置参数时,如PreparedStatement.setXXX()
  • JDBC Type → Java Type(获取结果时,如ResultSet.getXXX()

MyBatis 内置了大量 TypeHandler(如StringTypeHandler,IntegerTypeHandler),但面对复杂类型时,我们需要自定义。


二、实战场景一:MySQL JSON 字段 ↔ Java 对象

场景说明

用户表中有一个profile JSON字段,存储用户扩展信息:

CREATE TABLE tbl_user ( id INT PRIMARY KEY, username VARCHAR(50), profile JSON );

Java 实体:

public class User { private Integer id; private String username; private UserProfile profile; // 自定义对象 } public class UserProfile { private String avatar; private String city; // getter / setter }

目标:查询时自动将 JSON 字符串转为UserProfile;插入时反向转换。


步骤 1:引入 JSON 工具(如 Jackson)

<dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-databind</artifactId> </dependency>

步骤 2:编写自定义 TypeHandler

// JsonTypeHandler.java package com.charles.typehandler; import com.fasterxml.jackson.databind.ObjectMapper; import org.apache.ibatis.type.BaseTypeHandler; import org.apache.ibatis.type.JdbcType; import java.sql.*; public class JsonTypeHandler<T> extends BaseTypeHandler<T> { private static final ObjectMapper objectMapper = new ObjectMapper(); private Class<T> type; public JsonTypeHandler(Class<T> type) { if (type == null) throw new IllegalArgumentException("Type argument cannot be null"); this.type = type; } @Override public void setNonNullParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException { try { String json = objectMapper.writeValueAsString(parameter); ps.setString(i, json); // 存为 VARCHAR/TEXT } catch (Exception e) { throw new SQLException("Error converting " + type.getSimpleName() + " to JSON", e); } } @Override public T getNullableResult(ResultSet rs, String columnName) throws SQLException { return parseJson(rs.getString(columnName)); } @Override public T getNullableResult(ResultSet rs, int columnIndex) throws SQLException { return parseJson(rs.getString(columnIndex)); } @Override public T getNullableResult(CallableStatement cs, int columnIndex) throws SQLException { return parseJson(cs.getString(columnIndex)); } private T parseJson(String json) throws SQLException { if (json == null || json.isEmpty()) return null; try { return objectMapper.readValue(json, type); } catch (Exception e) { throw new SQLException("Error parsing JSON to " + type.getSimpleName(), e); } } }

✅ 继承BaseTypeHandler<T>,实现四个核心方法; ✅ 使用泛型支持任意 Java 对象; ✅ 异常统一包装为SQLException


步骤 3:在实体类中注册 TypeHandler

public class User { private Integer id; private String username; @Results({ @Result(property = "profile", column = "profile", typeHandler = JsonTypeHandler.class) }) private UserProfile profile; }

或更简洁地,在字段上使用@TypeHandler(MyBatis 3.4+):

public class User { // ... @TypeHandler(JsonTypeHandler.class) private UserProfile profile; }

🔔 注意:若使用 XML 映射,可在<result>标签中指定typeHandler


步骤 4:测试效果

@Test public void testJsonTypeHandler() { User user = new User(); user.setUsername("张三"); UserProfile profile = new UserProfile(); profile.setAvatar("avatar.jpg"); profile.setCity("深圳"); user.setProfile(profile); userMapper.insert(user); // 自动转为 JSON 字符串存入数据库 User saved = userMapper.selectById(user.getId()); System.out.println(saved.getProfile().getCity()); // 输出:深圳 }

✅ 成功!无需手动序列化/反序列化!


三、实战场景二:枚举存储(Enum ↔ Integer/String)

需求

订单状态:CREATED=0,PAID=1,SHIPPED=2

public enum OrderStatus { CREATED(0), PAID(1), SHIPPED(2); private final int code; OrderStatus(int code) { this.code = code; } public int getCode() { return code; } public static OrderStatus fromCode(int code) { for (OrderStatus s : values()) { if (s.code == code) return s; } throw new IllegalArgumentException("Invalid code: " + code); } }

自定义 EnumTypeHandler

public class OrderStatusTypeHandler extends BaseTypeHandler<OrderStatus> { @Override public void setNonNullParameter(PreparedStatement ps, int i, OrderStatus parameter, JdbcType jdbcType) throws SQLException { ps.setInt(i, parameter.getCode()); } @Override public OrderStatus getNullableResult(ResultSet rs, String columnName) throws SQLException { int code = rs.getInt(columnName); return rs.wasNull() ? null : OrderStatus.fromCode(code); } // ... 其他 getNullableResult 方法类似 }

Order实体中使用:

@TypeHandler(OrderStatusTypeHandler.class) private OrderStatus status;

💡 优势:数据库存整数,Java 用枚举,安全又语义清晰!


四、全局注册 TypeHandler(可选)

避免每个字段重复声明,可在mybatis-config.xml中全局注册:

<typeHandlers> <typeHandler handler="com.charles.typehandler.JsonTypeHandler" javaType="com.charles.entity.UserProfile" jdbcType="VARCHAR" /> </typeHandlers>

或在 Spring Boot 中通过配置类:

@Configuration public class MyBatisConfig { @Bean public SqlSessionFactory sqlSessionFactory(DataSource dataSource) throws Exception { SqlSessionFactoryBean factory = new SqlSessionFactoryBean(); factory.setDataSource(dataSource); // 注册全局 TypeHandler TypeHandlerRegistry registry = factory.getObject().getConfiguration().getTypeHandlerRegistry(); registry.register(UserProfile.class, new JsonTypeHandler<>(UserProfile.class)); return factory.getObject(); } }

五、注意事项 & 最佳实践

⚠️ 1. 线程安全

  • ObjectMapper是线程安全的(Jackson 2.8+),可共享;
  • 避免在 TypeHandler 中使用非线程安全的成员变量。

⚠️ 2. 异常处理

  • 必须捕获内部异常并转为SQLException,否则 MyBatis 无法正确回滚。

✅ 3. 性能

  • TypeHandler 在每次 SQL 执行时调用,避免做耗时操作;
  • 可缓存反射结果(如枚举 code 映射)。

🔄 4. 与 JSON 数据库类型兼容

  • MySQL 5.7+ 支持JSON类型,但 JDBC 驱动仍将其视为VARCHAR
  • 因此setString/getString完全适用。

六、总结:TypeHandler 应用场景速查

场景解决方案
JSON 字段 ↔ 对象自定义JsonTypeHandler
枚举 ↔ 数字/字符串自定义EnumTypeHandler
加密字段(如手机号)入库加密,出库解密
日期格式定制(如只存年月)自定义LocalDateTypeHandler
多值字段(如逗号分隔)转为List<String>

核心价值
“让数据库字段与 Java 对象自由对话,代码更干净,逻辑更内聚!”

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/15 4:03:40

WPS办公自动化新篇章:VBA插件7.1全面解析与实战指南

WPS办公自动化新篇章&#xff1a;VBA插件7.1全面解析与实战指南 【免费下载链接】最新版VBA插件7.1支持WPS 本仓库提供最新版VBA插件7.1的下载资源&#xff0c;该插件专为WPS设计&#xff0c;能够帮助用户在WPS中高效使用VBA功能 项目地址: https://gitcode.com/open-source-…

作者头像 李华
网站建设 2026/4/12 9:23:01

构建高性能量子计算容器的7个关键技术点(稀缺实战经验曝光)

第一章&#xff1a;量子计算容器化的核心挑战将量子计算与容器化技术结合&#xff0c;是推动量子应用工程化部署的关键路径。然而&#xff0c;由于量子计算本身的特殊性&#xff0c;其在容器化过程中面临诸多技术障碍&#xff0c;涉及资源管理、环境隔离和硬件交互等多个层面。…

作者头像 李华
网站建设 2026/4/2 7:16:31

测试资源分配的挑战与机遇

在软件测试领域&#xff0c;资源分配始终是项目成功的关键因素。传统方法依赖人工经验&#xff0c;往往面临测试环境、人力和时间资源的浪费&#xff0c;导致覆盖率不足或成本超支。随着人工智能和机器学习技术的兴起&#xff0c;智能优化为测试资源分配提供了新思路。本文旨在…

作者头像 李华
网站建设 2026/4/10 15:08:19

RPA 驱动自动化优先思维:打造高效组织的实践指南

在数字化转型深水区&#xff0c;“自动化优先” 正从前沿理念升级为企业主流管理思维。越来越多组织意识到&#xff0c;传统 “先标准化、后自动化” 的模式已难以适配快速变化的市场需求&#xff0c;而以自动化为核心重构业务流程&#xff0c;才能真正实现降本增效、敏捷应变。…

作者头像 李华
网站建设 2026/4/15 7:36:18

挥手点亮圣诞:AI 3D 魔法树教程

临近圣诞节&#xff0c;大家都在思考如何让今年更有仪式感&#xff1a;搞一棵圣诞树&#xff0c;还是干脆上一张滤镜海报&#xff1f;但如果你想让节日过得更「有科技味儿」&#xff0c;今年有个更酷的选择——一棵能听懂你手势、会展示你的照片、还能随你指挥聚散旋转的「3D C…

作者头像 李华
网站建设 2026/4/14 17:21:10

软件许可优化技术栈:动态资源池化+智能调度+合规审计融合

软件许可优化技术栈&#xff1a;动态资源池化智能调度合规审计融合一、为什么企业需要软件许可优化&#xff1f;在数字化转型的浪潮下&#xff0c;软件已经成为各行各业的核心资产&#xff0c;但随之而来的软件许可成本却不断攀升&#xff0c;成为很多企业成本控制的痛点。是在…

作者头像 李华