MyBatis Plus:自定义typeHandler类型处理器

目录

引言:关于TypeHandler

PostGreSQL:JSON数据类型

PostGreSQL数据库驱动:PGobject类

TypeHandler类型处理器

自定义类型处理器

类型处理器实现:PGJsonTypeHandler

注册类型处理器


引言:关于TypeHandler

        MyBatis Plus提供了丰富的类型处理器TypeHandler,用于实现用于 JavaType 与 JdbcType 之间的转换,用于 PreparedStatement 设置参数值和从 ResultSet 或 CallableStatement 中取出一个值。

        但是,内置的类型处理器在某些特殊情况难以满足实际开发需求。以下,以PostGreSQL数据库中的JSON类型为例,叙述自定义类型处理器,实现java.util.Map与PostGreSQL中JSON类型之间的转换。

PostGreSQL:JSON数据类型

        使用SQL语句构建JSON类型字段值是十分简单的,

-- 简单标量/基本值
-- 基本值可以是数字、带引号的字符串、true、false或者null
SELECT '5'::json;-- 有零个或者更多元素的数组(元素不需要为同一类型)
SELECT '[1, 2, "foo", null]'::json;-- 包含键值对的对象
-- 注意对象键必须总是带引号的字符串
SELECT '{"bar": "baz", "balance": 7.77, "active": false}'::json;-- 数组和对象可以被任意嵌套
SELECT '{"foo": [true, "bar"], "tags": {"a": 1, "b": null}}'::json;

        问题在于:如何将java中的Map类型转换为PG数据库中的JSON字段值。

PostGreSQL数据库驱动:PGobject类

        在PostGreSQL数据库驱动依赖包的org.postgresql.util工具包下,挖出来了一个名称为:PGobject的类,

        该类(PGobject)的源码描述信息如下:

PGobject is a class used to describe unknown types An unknown type is any type that is unknown by JDBC Standards.

概略含义:PGobject是一个用于描述一种非JDBC标准的未知类型。

        可以看到,它内置了很多子类,用于描述PostGreSQL中的Line、Polygon、Money等数据类型。

         一通分析之后,发现PGobject类的type、value属性是比较重要的(毕竟只提供了这两个属性),

        查看源码之后,解读type和value字段的含义:

type:object对象的类型,

value:object对象的值。

        再拿来和PGpolygon子类做一下对比,会发现:type属性其实就是一个String字符串,用于描述PostGreSQL内置数据类型的名称;value则是具体的值。那么,我们也可以借助这个PGobject类来描述自己的类型。

TypeHandler类型处理器

        TypeHandler类型处理器是MyBatis/MyBatisPlus很重要的一部分内容,日常开发中,从数据表到JavaBean,从JavaBean到数据表,两者之间的转换全靠这些内置的TypeHandler来实现。

        查看TypeHandler的源码,内容如下,

/*** @author Clinton Begin*/
public interface TypeHandler<T> {void setParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException;/*** Gets the result.** @param rs*          the rs* @param columnName*          Column name, when configuration <code>useColumnLabel</code> is <code>false</code>* @return the result* @throws SQLException*           the SQL exception*/T getResult(ResultSet rs, String columnName) throws SQLException;T getResult(ResultSet rs, int columnIndex) throws SQLException;T getResult(CallableStatement cs, int columnIndex) throws SQLException;}

        可以分为两类,其中:

①setParameter:是将JavaBean属性值转换为PostGreSQL支持的数据库字段值;

②3个getResult方法:熟悉原生JDBC编程的朋友应该都很眼熟,其实就是通过PreparedStatement执行完SQL之后,对ResultSet结果集进行按字段名columnName、按索引下标columnIndex,将PostGreSQL支持的数据库字段值,转换为JavaBean属性值的过程。

自定义类型处理器

        了解了上述内容之后,我们要自定义类型处理器,核心的工作就是:对TypeHandler接口提供的三个方式进行实现。

类型处理器实现:PGJsonTypeHandler

        以下给出一个Map集合类型到PostGreSQL-JSON数据类型之间的转换器实现类,相关部分可以直接查看代码注释,

/*** TypeHandler:类型处理器,用于 JavaType 与 JdbcType 之间的转换*  [1] 用于 PreparedStatement 设置参数值*  [2] 用于从 ResultSet CallableStatement 中取出一个值* PGJsonTypeHandler:处理Map集合类型与postgresql中JSON类型之间的转换*/@MappedTypes(value = {Object.class})
public class PGJsonTypeHandler<T> extends BaseTypeHandler<T> {//PGobject:PGobject is a class used to describe unknown types An unknown type is any type that is unknown by JDBC Standards.private static final PGobject pgObject = new PGobject();/*** 插入时设置参数类型* @param preparedStatement SQL预编译对象* @param i 需要赋值的索引位置(相当于在JDBC中对占位符的位置进行赋值)* @param parameter 索引位置i需要赋的值(原本要给这个位置赋的值,在setNonNullParameter方法中主要解决的问题就是将这个自定义类型变成数据库认识的类型)* @param jdbcType jdbc的类型* @throws SQLException*/@Overridepublic void setNonNullParameter(PreparedStatement preparedStatement, int i, T parameter, JdbcType jdbcType) throws SQLException {if (preparedStatement != null) {System.out.println("setNonNullParameter::"+parameter.toString());pgObject.setType("json");pgObject.setValue(JSON.toJSONString(parameter));preparedStatement.setObject(i, pgObject);}}/*** 转换为T类型实例* @param object 参数字符串({"name":"张三","age":18,"address":"北京市朝阳区"})或者null空值* @return T实例或者参数*/private T parseAsTInstance(Object object){//判空if(object == null){return null;}//处理字符串{"name":"张三","age":18,"address":"北京市朝阳区"}-转换为Map类型Map map = JSON.parseObject(object.toString(), Map.class);return (T)map;}/*** 获取时转换回的自定义类型* @param resultSet 结果集* @param s 列名称* @return Bean对象*/@Overridepublic T getNullableResult(ResultSet resultSet, String s) throws SQLException {Object object = resultSet.getObject(s);return parseAsTInstance(object);}/*** 获取时转换回的自定义类型* @param resultSet 结果集* @param i 列索引*/@Overridepublic T getNullableResult(ResultSet resultSet, int i) throws SQLException {Object object = resultSet.getObject(i);return parseAsTInstance(object);}/*** 获取时转换回的自定义类型* @param callableStatement 结果集* @param i 列索引*/@Overridepublic T getNullableResult(CallableStatement callableStatement, int i) throws SQLException {Object object = callableStatement.getObject(i);return parseAsTInstance(object);}
}

注册类型处理器

        要注册类型处理器,可以进行全局配置:typeHandlersPackage属性

        也可以在局部直接使用,例如:

package com.example.soil_backend.model.entity;import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableName;
import com.baomidou.mybatisplus.annotation.IdType;import java.time.LocalDateTime;import com.baomidou.mybatisplus.annotation.TableId;import java.io.Serializable;
import java.util.Map;import com.example.soil_backend.model.TbBase;
import com.example.soil_backend.typehandler.PGJsonbTypeHandler;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.experimental.Accessors;/*** 配置模板** @author XiMumu* @since 2024-02-18*/
@Data
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@TableName(value = "tb_template", autoResultMap = true)
public class TbTemplate extends TbBase implements Serializable {private static final long serialVersionUID = -2288822970791637243L;/*** 主键ID*/@TableId(value = "id", type = IdType.AUTO)private Integer id;/*** 调查模板名称*/@TableField(value = "name")private String name;/*** 模板配置JSON*/@TableField(value = "config",typeHandler = PGJsonbTypeHandler.class)private Map config;/*** 是否删除:1-使用中;2-已删除*/@TableField(value = "del_flag")private Integer delFlag;}

      

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

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

相关文章

Java项目:22 Javaweb实现的学生宿舍管理系统

作者主页&#xff1a;舒克日记 简介&#xff1a;Java领域优质创作者、Java项目、学习资料、技术互助 文中获取源码 项目介绍 Javaweb实现的学生宿舍管理系统采用jspservlet技术实现了如下功能模块&#xff0c;分别是宿舍管理员管理、学生管理、宿舍楼管理、缺勤记录、修改密码…

基于springboot+vue的B2B平台的医疗病历交互系统(前后端分离)

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、阿里云专家博主、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战&#xff0c;欢迎高校老师\讲师\同行交流合作 ​主要内容&#xff1a;毕业设计(Javaweb项目|小程序|Pyt…

哪款护眼台灯比较耐用?2024年五款甄选高评分榜单好物!

作为一位专业家电测评博主&#xff0c;经常会有粉丝们的询问&#xff1a;究竟哪个牌子的护眼台灯好用又护眼&#xff1f;许多人在购买时由于信息不对等&#xff0c;结果选择的产品往往存在诸如光照不舒适、损伤眼睛等问题。为了让大家都能精准找到能够保护眼睛的台灯&#xff0…

大工程 从0到1 数据治理 数仓篇(sample database classicmodels _No.7)

大工程 从0到1 数据治理 之数仓篇 我这里还是sample database classicmodels为案列&#xff0c;可以下载&#xff0c;我看 网上还没有类似的 案列&#xff0c;那就 从 0-1开始吧&#xff01; 提示&#xff1a;写完文章后&#xff0c;目录可以自动生成&#xff0c;如何生成可参…

【安卓基础3】Activity(一)

&#x1f3c6;作者简介&#xff1a;|康有为| &#xff0c;大四在读&#xff0c;目前在小米安卓实习&#xff0c;毕业入职 &#x1f3c6;本文收录于 安卓学习大全&#xff0c;欢迎关注 &#x1f3c6;安卓学习资料推荐&#xff1a; 视频&#xff1a;b站搜动脑学院 视频链接 &…

WEB甘特图选型

1.火车侠 GitHub - w1301625107/Vue-Gantt-chart: 使用Vue做数据控制的Gantt图表 优点&#xff1a;有小时选项 缺点&#xff1a;不支持拖拽 2.火车侠二号 GitHub - liyang5945/vue-drag-gantt-chart: Vue拖拽甘特图 a draggable gantt-chart with vue 有点&#xff1a;实现了…

2012及其以上系统修改服务器密码指南

修改服务器密码指南,目前介绍两种不同的方案 方法一 指令式 winR键 弹出运行框里输入 cmd 点击确认或者右下角开始程序里面的点开运行 2.在弹出框里手动输入以下一组文字&#xff1a;net user administrator 123456 框内无法粘贴 需要手动输入 其中administrator 是用…

MacBook安装Ansible

MacBook安装Ansible Ansible介绍 ansible是新出现的自动化运维工具&#xff0c;基于Python开发&#xff0c;集合了众多运维工具&#xff08;puppet、chef、func、fabric&#xff09;的优点&#xff0c;实现了批量系统配置、批量程序部署、批量运行命令等功能。 ansible是基于…

PreMaint CMS系统:数字化驱动起重机远程管理的智能未来

在港口起重机领域&#xff0c;数字化解决方案正成为提高效率、降低停机时间的关键。PreMaint CMS作为起重机核心可视化系统&#xff0c;融合了操作人员、维护团队和管理者的需求&#xff0c;通过智能化的方式&#xff0c;将复杂的数据转化为简洁的信息&#xff0c;实现对起重机…

网络图谱构建系统目前已实现的功能

一.移动智能终端&#xff1a; 1.主页面&#xff1a; 地图层调用百度地图api。要在百度地图开发社区申请密钥和服务。 界面中卡片&#xff0c;悬浮按钮&#xff0c;上标题栏都采用谷歌公司material desgin设计风格。 2.标题栏&#xff1a; 采用toolbar&#xff0c;可以应用程…

Python习题详解

练习&#xff1a; 1&#xff0c;计算100以内奇数的和 #计算100以内所有奇数的和 sum 0 # n 1 # while n < 100: # # sum sum n # sum n # # n n 2 # n 2 # print(sum) n 99 #求偶数时n 100 while n > 0: sum n # n n - 2 n - 2 print(sum)2&#xff0c;打印…

yolov5-tracking-xxxsort yolov5融合六种跟踪算法(二)--目标识别

本次开源计划主要针对大学生无人机相关竞赛的视觉算法开发。 开源代码仓库链接&#xff1a;https://github.com/zzhmx/yolov5-tracking-xxxsort.git 先按照之前的博客配置好环境&#xff1a; yolov5-tracking-xxxsort yolov5融合六种跟踪算法&#xff08;一&#xff09;–环境配…