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.pswp.cn/news/895765.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/895765.shtml
英文地址,請注明出處:http://en.pswp.cn/news/895765.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Qt開發①Qt的概念+發展+優點+應用+使用

目錄 1. Qt的概念和發展 1.1 Qt的概念 1.2 Qt 的發展史&#xff1a; 1.3 Qt 的版本 2. Qt 的優點和應用 2.1 Qt 的優點&#xff1a; 2.2 Qt 的應用場景 2.3 Qt 的應用案例 3. 搭建 Qt 開發環境 3.1 Qt 的開發工具 3.2 Qt SDK 的下載和安裝 3.3 Qt 環境變量配置和使…

mac安裝Pyspark并連接Mysql

安裝Scala, apache-spark, Hadoop brew install scala brew install apache-spark brew install hadoop pip install pyspark注意不要自己另外安裝jdk, 會造成版本對不上報錯。因為安裝apache-spark的過程中會自動安裝openjdk。 配置環境變量 JAVA_HOME/opt/homebrew/Cellar…

【Go語言快速上手】第二部分:Go語言進階之網絡編程

文章目錄 前言&#xff1a;網絡編程一、TCP/UDP 編程&#xff1a;net 包的使用1. TCP 編程1.1 TCP 服務器1.2 TCP 客戶端 2. UDP 編程2.1 UDP 服務器2.2 UDP 客戶端 二、HTTP 編程&#xff1a;net/http 包的使用&#xff0c;編寫 HTTP 服務器和客戶端2.1 HTTP 服務器2.2 HTTP 客…

王炸 用AI+飛書 分解 一鍵生成 項目計劃表模版

效果圖&#xff1a; 各字段設置&#xff1a; 以下是一個使用 AI&#xff08;DeepSeeker&#xff09; 飛書多維表格分解項目待辦模板的示例&#xff0c;你可以根據實際情況進行調整和優化&#xff1a; 列表中需要選擇對象&#xff0c;且選擇輸出結果&#xff08;記得控制字符長度…

從月牙定理看古希臘數學的奇妙突破

文章目錄 每日一句正能量前言古希臘人的 “化圓為方” 之夢&#xff08;一&#xff09;幾何作圖的基本規則&#xff08;二&#xff09;化圓為方問題的起源與發展&#xff08;三&#xff09;化圓為方的意義 月牙面積定理的誕生&#xff08;一&#xff09;希波克拉底的生平與成就…

實戰:vLLM多機多卡部署大模型

兩臺服務器 1. Docker容器中使用GPU 必須確保已安裝并配置 NVIDIA Docker。你可以安裝 nvidia-docker 來確保 GPU 驅動能夠被 Docker 使用 #安裝 nvidia-docker&#xff1a; sudo apt-get install nvidia-docker2#然后重啟 Docker&#xff1a; sudo systemctl restart docke…

LLM中種子(Seed)作用是什么:多樣性

LLM中種子(Seed)作用是什么:多樣性 目錄 LLM中種子(Seed)作用是什么:多樣性作用舉例不同種子的區別設置不同種子的原因在LLM(大語言模型)中,種子(Seed)用于初始化隨機數生成器,發揮著確保結果可重復性的關鍵作用,具體如下: 作用 當大語言模型生成文本時,很多操…

neo4j二進制部署

neo4j二進制部署 下載所需組件 jdk 17 neo4j 5.2.0 配置文件 server.default_listen_address0.0.0.0環境變量 export JAVA_HOME/usr/local/jdk-17.0.13 export CLASSPATH.:${JAVA_HOME}/jre/lib/rt.jar:${JAVA_HOME}/lib/dt.jar:${JAVA_HOME}/lib/tools.jar export PATH$…

Win10環境使用零訊ZeroNews內網穿透實現Deepseek對外服務

Win10環境使用零訊ZeroNews內網穿透實現Deepseek對外服務 前言 之前筆者已經在Win10環境搭建好了Ollama、DeepSeek、Open WebUI、Dify等組件&#xff0c;成功實現了私有化部署及內網訪問&#xff1a; https://lizhiyong.blog.csdn.net/article/details/145505686 https://l…

spconv 安裝測試

pip install spconv 報錯: File "/usr/local/lib/python3.10/dist-packages/torch/nn/modules/module.py", line 1736, in _wrapped_call_impl return self._call_impl(*args, **kwargs) File "/usr/local/lib/python3.10/dist-packages/torch/nn/modules/mod…

MySQL 查詢緩存技術深度解析

在現代數據庫管理系統中&#xff0c;查詢性能優化是提升應用響應速度和用戶體驗的關鍵環節。MySQL 作為一款廣泛使用的開源關系型數據庫&#xff0c;提供了查詢緩存功能&#xff0c;用于緩存查詢結果&#xff0c;從而在后續相同的查詢請求時能夠快速返回結果&#xff0c;減少數…

halcon 條形碼、二維碼識別、opencv識別

一、條形碼 函數介紹 create_bar_code_model * 1.創建條碼讀取器的模板 * 參數一&#xff1a;通用參數的名稱&#xff0c;針對條形碼模型進行調整。默認值為空 * 參數二&#xff1a;針對條形碼模型進行調整 * 參數三&#xff1a;條形碼模型的句柄。 create_bar_code_model (…

一個簡潔高效的Flask用戶管理示例

Flask-Login 是 Flask 的用戶管理擴展&#xff0c;提供 用戶身份驗證、會話管理、權限控制 等功能。 適用于&#xff1a; ? 用戶登錄、登出 ? 記住用戶&#xff08;“記住我” 功能&#xff09; ? 限制未登錄用戶訪問某些頁面 ? 用戶會話管理 1. 安裝 Flask-Login pi…

HashSet 的底層原理(簡單易懂)

在 Java 集合框架中&#xff0c;HashSet 是一個非常常用的集合類&#xff0c;它提供了快速的元素查找和插入操作。那么&#xff0c;HashSet 的底層是如何實現這些高效操作的呢&#xff1f;本文將深入探討 HashSet 的底層原理。 一、HashSet 的基本概念 HashSet 是基于哈希表的…

【學習資源】時間序列數據分析方法(2)-mWDN和AutoEncoder

接著上次的【學習資源】時間序列數據分析方法&#xff08;1&#xff09;-CSDN博客&#xff0c;本次介紹mWDN和AutoEncoder 解決時序數據分類的方法。介紹模型原理、應用場景和參考代碼。也從模型性能、訓練效率、模型復雜度、計算復雜度、可解釋性、適應性和泛化能力、健壯性、…

[LeetCode力扣hot100]-鏈表

相交鏈表 160. 相交鏈表 - 力扣&#xff08;LeetCode&#xff09; 思路就是遍歷兩個鏈表&#xff0c;有相同的部分就可以視為相交。 但是長度不一樣&#xff0c;比如兩個會相交的鏈表&#xff0c;headA 的長度為 a c&#xff0c;headB 的長度為 b c&#xff0c;其中 c 是公…

JAVA EE初階 - 預備知識(四)

一、API API 即應用程序編程接口&#xff08;Application Programming Interface&#xff09;&#xff0c;是一組定義、協議和工具&#xff0c;用于不同軟件組件、應用程序或系統之間進行交互和通信。以下從多個方面詳細介紹 API&#xff1a; 基本概念 接口規范&#xff1a;A…

【TI C2000】F28002x的系統延時、GPIO配置及SCI(UART)串口發送、接收

【TI C2000】F28002x的系統延時、GPIO配置及SCI&#xff08;UART&#xff09;串口發送、接收 文章目錄 系統延時GPIO配置GPIO輸出SCI配置SCI發送、接收測試附錄&#xff1a;F28002x開發板上手、環境配置、燒錄及TMS320F280025C模板工程建立F28002x敘述燒錄SDK庫文件說明工程建…

親測有效!使用Ollama本地部署DeepSeekR1模型,指定目錄安裝并實現可視化聊天與接口調用

文章目錄 一、引言二、準備工作&#xff08;Ollama 工具介紹與下載&#xff09;2.1 Ollama介紹2.2 Ollama安裝 三、指定目錄安裝 DeepSeek R1四、Chatbox 可視化聊天搭建4.1 Chatbox下載安裝4.2 關聯 DeepSeek R1 與 Chatbox 的步驟 五、使用 Ollama 調用 DeepSeek 接口5.1 請求…

期權隱含波動率是什么意思?

財順小編本文主要介紹期權隱含波動率是什么意思&#xff1f;期權隱含波動率&#xff08;Implied Volatility&#xff09;是根據當前期權市場價格&#xff0c;利用期權定價模型&#xff08;如Black-Scholes模型&#xff09;推導出的關于合約標的理論上的價格波動率。它反映了市場…