mybatis使用typeHandler实现类型转换

使用mybatis作为操作数据库的orm框架,操作基本数据类型时可以通过内置的类型处理器完成java数据类型和数据库类型的转换,但是对于扩展的数据类型要实现与数据库类型的转换就需要自定义类型转换器完成,比如某个实体类型存储到数据库,可以转换为json字符串存储,读取数据时再转换为对应的实体类。
在mybatis中可以有两种方式实现上面的方案:
一、直接继承mybatis框架提供的 org.apache.ibatis.type.BaseTypeHandler 完成数据类型转换;
二、如果项目引入了mybatis-plus,也可以继承 com.baomidou.mybatisplus.extension.handlers.AbstractJsonTypeHandler 实现数据类型转换。
接下来分别介绍上面两种方案的实现方式。
首先在数据库中创建一个表用于测试数据存取:

CREATE TABLE `demo_data`  (`id` int NOT NULL AUTO_INCREMENT,`detail` json NULL,`create_time` datetime NULL,PRIMARY KEY (`id`)
);
一、mybatis框架实现类型转换

使用mybatis实现类型转换,首先要自定义一个handler继承自基础的handler,再将自定义的handler注入到字段的typeHandler中就实现了类型转换:

自定义handler:

import org.apache.ibatis.type.BaseTypeHandler;
import org.apache.ibatis.type.JdbcType;
import org.apache.ibatis.type.MappedJdbcTypes;import java.sql.CallableStatement;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;/*** @Author xingo* @Date 2025/2/6*/
@MappedJdbcTypes({JdbcType.VARCHAR, JdbcType.CHAR})
public class DetailTypeHandler extends BaseTypeHandler<DemoData.DetailInfo> {@Overridepublic void setNonNullParameter(PreparedStatement ps, int i, DemoData.DetailInfo parameter, JdbcType jdbcType) throws SQLException {ps.setString(i, JacksonUtils.toJSONString(parameter));}@Overridepublic DemoData.DetailInfo getNullableResult(ResultSet rs, String columnName) throws SQLException {return JacksonUtils.parseObject(rs.getString(columnName), DemoData.DetailInfo.class);}@Overridepublic DemoData.DetailInfo getNullableResult(ResultSet rs, int columnIndex) throws SQLException {return JacksonUtils.parseObject(rs.getString(columnIndex), DemoData.DetailInfo.class);}@Overridepublic DemoData.DetailInfo getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {return JacksonUtils.parseObject(cs.getString(columnIndex), DemoData.DetailInfo.class);}
}

上面就实现了java类型与数据库类型的对应关系,就是将实体类中的java对象与数据库中的字符串类型自动转换:

import lombok.Data;import java.io.Serializable;
import java.time.LocalDateTime;/*** @Author xingo* @Date 2025/2/6*/
@Data
public class DemoData implements Serializable {private Integer id;private DemoData.DetailInfo detail;private LocalDateTime createTime;@Datapublic static class DetailInfo implements Serializable {private String name;private Integer age;private LocalDateTime dateTime;}
}

接下来就是在编写的xml文件中将刚刚自定义的handler和实体类信息完成对应关系:

<?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="org.xingo.demo.DemoDataMapper"><resultMap id="demoData" type="org.xingo.demo.DemoData"><id property="id" column="id"/><result property="detail" column="detail" typeHandler="org.xingo.demo.DetailTypeHandler"/><result property="createTime" column="create_time"/></resultMap><insert id="insertDemoData" useGeneratedKeys="true" keyProperty="id">insert into demo_data(detail, create_time)values (#{detail, typeHandler=org.xingo.demo.DetailTypeHandler}, #{createTime})</insert><update id="updateDemoData">update demo_dataset detail=#{detail, typeHandler=org.xingo.demo.DetailTypeHandler}create_time=#{createTime}where id=#{id}</update><select id="findDemoDataById" resultMap="demoData">select *from demo_datawhere id=#{id}</select>
</mapper>

xml对应的接口:

import org.xingo.demo.DemoData;/*** @Author xingo* @Date 2025/2/6*/
public interface DemoDataMapper {void insertDemoData(DemoData data);void updateDemoData(DemoData data);DemoData findDemoDataById(Integer id);
}

上面的几步就实现了自定义数据类型与数据库中字符串类型的转换,测试上面接口可以完成数据的存取:请添加图片描述

二、mybatis-plus框架实现类型转换

使用mybatis实现自定义类型与数据库类型的转换相对来说还是有一点繁琐,如果在项目中引入了mybatis-plus,那么就可以减少xml文件的编写,直接在实体类的字段上添加注解完成xml文件的内容。

使用mybatis-plus实现类型转换首先也是自定义handler类:

import com.baomidou.mybatisplus.extension.handlers.AbstractJsonTypeHandler;
import org.apache.ibatis.type.JdbcType;
import org.apache.ibatis.type.MappedJdbcTypes;
import org.apache.ibatis.type.MappedTypes;/*** @Author xingo* @Date 2025/2/6*/
@MappedTypes({DemoData.DetailInfo.class})
@MappedJdbcTypes({JdbcType.VARCHAR, JdbcType.CHAR})
public class DetailTypeHandler extends AbstractJsonTypeHandler<DemoData.DetailInfo> {@Overrideprotected DemoData.DetailInfo parse(String json) {return JacksonUtils.parseObject(json, DemoData.DetailInfo.class);}@Overrideprotected String toJson(DemoData.DetailInfo detail) {return JacksonUtils.toJSONString(detail);}
}

映射主要是通过实体类的注解完成的:

import com.baomidou.mybatisplus.annotation.*;
import lombok.Data;import java.io.Serializable;
import java.time.LocalDateTime;/*** @Author xingo* @Date 2025/2/6*/
@Data
@TableName(value = "demo_data", autoResultMap = true)
public class DemoData implements Serializable {@TableId(type = IdType.AUTO)private Integer id;@TableField(typeHandler = DetailTypeHandler.class)private DemoData.DetailInfo detail;@TableField(fill = FieldFill.INSERT)private LocalDateTime createTime;@Datapublic static class DetailInfo implements Serializable {private String name;private Integer age;private LocalDateTime dateTime;}
}

mapper接口只需要继承mybatis-plus提供的基础mapper就可以:

import com.baomidou.mybatisplus.core.mapper.BaseMapper;/*** @Author xingo* @Date 2025/2/6*/
public interface DemoDataMapper extends BaseMapper<DemoData> {
}

通过上面的定义,所有基于mybatis-plus提供的增改查操作都可以完成字段类型转换。请添加图片描述
测试上面的内容在数据库中产生的数据:
请添加图片描述

附:jackson工具类

import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.*;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import com.fasterxml.jackson.datatype.jdk8.Jdk8Module;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateDeserializer;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalTimeDeserializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateSerializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalTimeSerializer;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;import java.io.IOException;
import java.text.SimpleDateFormat;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.LocalTime;
import java.time.format.DateTimeFormatter;
import java.util.List;
import java.util.Locale;
import java.util.TimeZone;/*** json工具** @Author xingo* @Date 2025/2/6*/
@Slf4j
public class JacksonUtils {private static final ObjectMapper OBJECT_MAPPER = new ObjectMapper();static {// Long类型处理,避免前端处理长整型时精度丢失SimpleModule module1 = new SimpleModule();module1.addSerializer(Long.class, ToStringSerializer.instance);module1.addSerializer(Long.TYPE, ToStringSerializer.instance);JavaTimeModule module2 = new JavaTimeModule();// java8日期处理module2.addSerializer(LocalDateTime.class, new LocalDateTimeSerializer(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));module2.addSerializer(LocalDate.class, new LocalDateSerializer(DateTimeFormatter.ofPattern("yyyy-MM-dd")));module2.addSerializer(LocalTime.class, new LocalTimeSerializer(DateTimeFormatter.ofPattern("HH:mm:ss")));module2.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));module2.addDeserializer(LocalDate.class, new LocalDateDeserializer(DateTimeFormatter.ofPattern("yyyy-MM-dd")));module2.addDeserializer(LocalTime.class, new LocalTimeDeserializer(DateTimeFormatter.ofPattern("HH:mm:ss")));OBJECT_MAPPER// 添加modules.registerModules(module1, module2, new Jdk8Module())// 日期类型不转换为时间戳.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false).configure(SerializationFeature.WRITE_DURATIONS_AS_TIMESTAMPS, false)// 反序列化的时候如果多了其他属性,不抛出异常.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false)// 如果是空对象的时候,不抛异常.configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false)// 空对象不序列化.setSerializationInclusion(JsonInclude.Include.NON_NULL)// 日期格式化.setDateFormat(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"))// 设置时区.setTimeZone(TimeZone.getTimeZone("GMT+8"))// 驼峰转下划线// .setPropertyNamingStrategy(PropertyNamingStrategies.SNAKE_CASE)// 语言.setLocale(Locale.SIMPLIFIED_CHINESE);}/*** 反序列化对象*/public static <T> T parseObject(String json, Class<T> clazz) {if (json == null) {return null;}try {return OBJECT_MAPPER.readValue(json, clazz);} catch (JsonProcessingException e) {e.printStackTrace();}return null;}/*** 反序列化对象*/public static JsonNode parseObject(String json) {if (json == null) {return null;}try {return OBJECT_MAPPER.readTree(json);} catch (JsonProcessingException e) {e.printStackTrace();}return null;}/*** 反序列化对象*/public static <T> T parseObject(String json, TypeReference<T> type) {if (json == null) {return null;}try {return OBJECT_MAPPER.readValue(json, type);} catch (JsonProcessingException e) {e.printStackTrace();}return null;}/*** 反序列化对象*/public static <T> T parseObject(byte[] bytes, TypeReference<T> type) {if (bytes == null) {return null;}try {return OBJECT_MAPPER.readValue(bytes, type);} catch (Exception e) {e.printStackTrace();}return null;}/*** 反序列化对象*/public static <T> T parseObject(JsonNode jsonNode, Class<T> clazz) {return jsonNode == null ? null : OBJECT_MAPPER.convertValue(jsonNode, clazz);}/*** 反序列化列表*/public static <T> List<T> parseArray(String json, Class<T> clazz) {if (json == null) {return null;}try {JavaType javaType = OBJECT_MAPPER.getTypeFactory().constructParametricType(List.class, clazz);return OBJECT_MAPPER.treeToValue(OBJECT_MAPPER.readTree(json), javaType);} catch (Exception e) {e.printStackTrace();}return null;}/*** 反序列化列表*/public static <T> List<T> parseArray(JsonNode json, Class<T> clazz) {try {JavaType javaType = OBJECT_MAPPER.getTypeFactory().constructParametricType(List.class, clazz);return json == null ? null : OBJECT_MAPPER.treeToValue(json, javaType);} catch (JsonProcessingException e) {log.warn(e.getLocalizedMessage());return null;}}/*** 写为json串*/public static String toJSONString(Object obj) {if (obj == null) {return null;}if (obj instanceof String) {return (String) obj;}try {return OBJECT_MAPPER.writeValueAsString(obj);} catch (JsonProcessingException e) {e.printStackTrace();}return null;}/*** 写为字节数组*/public static byte[] toJSONBytes(Object obj) {if (obj == null) {return null;}try {return OBJECT_MAPPER.writeValueAsBytes(obj);} catch (Exception e) {e.printStackTrace();}return null;}/*** 获取jackson对象*/public static ObjectMapper getObjectMapper() {return OBJECT_MAPPER;}/*** 美化输出json格式*/public static String pretty(String json) throws IOException {return StringUtils.isBlank(json) ? json : pretty(JacksonUtils.getObjectMapper().readTree(json));}/*** 美化输出json格式*/public static String pretty(JsonNode jsonNode) throws IOException {return null == jsonNode ? "" : JacksonUtils.getObjectMapper().writerWithDefaultPrettyPrinter().writeValueAsString(jsonNode);}/*** 对象转json*/public static JsonNode toJsonNode(Object obj) {if (obj instanceof String) {return parseObject((String) obj, JsonNode.class);}return obj == null ? null : OBJECT_MAPPER.convertValue(obj, JsonNode.class);}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/20511.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于微信小程序的宿舍报修管理系统设计与实现,SpringBoot(15500字)+Vue+毕业论文+指导搭建视频

运行环境 jdkmysqlIntelliJ IDEAmaven3微信开发者工具 项目技术SpringBoothtmlcssjsjqueryvue2uni-app 宿舍报修小程序是一个集中管理宿舍维修请求的在线平台&#xff0c;为学生、维修人员和管理员提供了一个便捷、高效的交互界面。以下是关于这些功能的简单介绍&#xff1a; …

STM32 HAL库 UART串口发送数据实验

一、实验目标 通过STM32的UART串口发送字符串数据到PC端串口调试助手&#xff0c;验证通信功能。 二、硬件准备 主控芯片&#xff1a;STM32F103C8T6。 串口模块&#xff1a;USB转TTL模块。 接线说明&#xff1a; STM32的USART1_TX&#xff08;PA9&#xff09; → USB-TTL模…

一.AI大模型开发-初识机器学习

机器学习基本概念 前言 本文主要介绍了深度学习基础&#xff0c;包括机器学习、深度学习的概念&#xff0c;机器学习的两种典型任务分类任务和回归任务&#xff0c;机器学习中的基础名词解释以及模型训练的基本流程等。 一.认识机器学习 1.人工智能和机器学习 人工智能&am…

冒险岛079 V8 整合版源码搭建教程+IDEA启动

今天教大家来部署下一款超级怀旧游戏冒险岛&#xff0c;冒险岛源码是开源的&#xff0c;但是开源的代码会有各种&#xff0c;本人进行了加工整合&#xff0c;并且用idea进行了启动测试&#xff0c;经过修改后没有任何问题。 启动截图 后端控制台 前端游戏界面 声明 冒险岛源码…

【操作系统】操作系统概述

操作系统概述 1.1 操作系统的概念1.1.1 操作系统定义——什么是OS&#xff1f;1.1.2 操作系统作用——OS有什么用&#xff1f;1.1.3 操作系统地位——计算机系统中&#xff0c;OS处于什么地位&#xff1f;1.1.4 为什么学操作系统&#xff1f; 1.2 操作系统的历史1.2.1 操作系统…

单元测试junit5

一、idea 安装自动化生成插件jcode5 安装可能不成功&#xff0c;尝试多次安装&#xff1b; 安装成功后&#xff0c;重启idea&#xff0c;再次确认安装是否成功&#xff1b; 二、在需要生成单元测试代码的模块的pom中引入依赖 ......<parent><groupId>org.springf…

mysql主从配置(2025)

一、配置主服务器 编辑主mysql配置文件my.cnf&#xff08;vim /etc/my.cnf&#xff09;&#xff0c;在[mysqld]下添加 [mysqld] # 配置主ID,必须在所有参与主从复制的数据库保证唯一 server-id1 # 打开二进制日志 log-bin/var/lib/mysql/mysql-bin.log # 只允许同步ente_dat…

6.2.图的存储结构-邻接矩阵法

一.邻接矩阵法存储不带权图&#xff1a; 结点不带权值&#xff1a; 1.左图的无向图中&#xff0c;A到B直达的有一条路&#xff0c;所以A行B列的值为1&#xff1b; 左图的无向图中&#xff0c;A到F没有直达的路&#xff0c;所以A行F列的值为0&#xff1b; 结论&#xff1a;无…

1-知识图谱-概述和介绍

知识图谱&#xff1a;浙江大学教授 陈华军 知识图谱 1课时 http://openkg.cn/datasets-type/ 知识图谱的价值 知识图谱是有什么用&#xff1f; 语义搜索 问答系统 QA问答对知识图谱&#xff1a;结构化图 辅助推荐系统 大数据分析系统 自然语言理解 辅助视觉理解 例…

【C语言】C语言 食堂自动化管理系统(源码+数据文件)【独一无二】

&#x1f449;博__主&#x1f448;&#xff1a;米码收割机 &#x1f449;技__能&#x1f448;&#xff1a;C/Python语言 &#x1f449;专__注&#x1f448;&#xff1a;专注主流机器人、人工智能等相关领域的开发、测试技术。 【C语言】C语言 食堂自动化管理系统&#xff08;源…

C#之上位机开发---------C#通信库及WPF的简单实践

〇、上位机&#xff0c;分层架构 界面层 要实现的功能&#xff1a; 展示数据 获取数据 发送数据 数据层 要实现的功能&#xff1a; 转换数据 打包数据 存取数据 通信层 要实现的功能&#xff1a; 打开连接 关闭连接 读取数据 写入数据 实体类 作用&#xff1a; 封装数据…

网络编程(24)——实现带参数的http-get请求

文章目录 二十四、day241. char 转为16进制2. 16进制转为 char3. URL 编码函数4. URL 解码函数5. 实现 get 请求参数的解析6. 测试 二十四、day24 我们在前文通过beast实现了http服务器的简单搭建&#xff0c;但是有很多问题我们并没有解决。 在前文中&#xff0c;我们的 get…

机器学习_18 K均值聚类知识点总结

K均值聚类&#xff08;K-means Clustering&#xff09;是一种经典的无监督学习算法&#xff0c;广泛应用于数据分组、模式识别和降维等领域。它通过将数据划分为K个簇&#xff0c;使得簇内相似度高而簇间相似度低。今天&#xff0c;我们就来深入探讨K均值聚类的原理、实现和应用…

LeetCode1287

LeetCode1287 目录 题目描述示例思路分析代码段代码逐行讲解复杂度分析总结的知识点整合总结 题目描述 给定一个非递减的整数数组 arr&#xff0c;其中有一个元素恰好出现超过数组长度的 25%。请你找到并返回这个元素。 示例 示例 1 输入: arr [1, 2, 2, 6, 6, 6, 6, 7,…

恒创科技:如何重新启动 Windows 服务器

重新启动 Windows 服务器对于应用更新、解决问题和维护系统性能至关重要。定期重新启动有助于确保服务器运行最新软件、解决冲突并清除临时文件。本教程将介绍如何使用不同的方法重新启动 Windows 服务器。 注意&#xff1a;重新启动服务器之前保存所有工作&#xff0c;以避免丢…

Django ModelForm使用(初学)

1.目的是根据员工表字段&#xff0c;实现一个新增员工的数据填写页面 2.在views.py文件中按下面的格式写 定义 ModelForm 类&#xff1a;UserModelForm &#xff08;自己命名的类名&#xff09;使用时需要导入包 定义视图函数&#xff1a;user_model_form_add&#xff08;在函…

华为固态电池引发的思索

华为固态电池真牛&#xff01; 超长续航&#xff1a;单次充电即可行驶3000公里 极速充电&#xff1a;五分钟内充满80% 极致安全&#xff1a;不可燃、不漏液 长寿命设计&#xff1a;循环寿命达10000次以上 如上是华为电池展示的优势项&#xff0c;每一条都让我们心动不已。…

美信监控易:运维新时代,守护数据安全

在 2025 年这个科技飞速发展的时代&#xff0c;数据安全已成为各行业关注的焦点。随着云计算、大数据、物联网等技术的不断推进&#xff0c;运维数据的保护面临着新的挑战与要求。美信时代公司的美信监控易运维管理软件&#xff0c;以其卓越的功能、特性和竞争力&#xff0c;为…

个人博客5年回顾

https://huangtao01.github.io/ 五年前&#xff0c;看程序羊的b站视频做的blog&#xff0c;受限于网络&#xff0c;只能单向学习&#xff0c;没有人指导与监督&#xff0c;从来没有想过&#xff0c;有没有什么问题&#xff1f; 一、为什么要做个人博客&#xff1f; 二、我是怎么…

Unity合批处理优化内存序列帧播放动画

Unity合批处理序列帧优化内存 介绍图片导入到Unity中的处理Unity中图片设置处理Unity中图片裁剪 创建序列帧动画总结 介绍 这里是针对Unity序列帧动画的优化内容&#xff0c;将多个图片合批处理然后为了降低Unity的内存占用&#xff0c;但是相对的质量也会稍微降低。可自行进行…