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

java 中 List<T> 类型数据在 postgreSql 数据库中存储

一 属性添加注解

  1. 在类上面添加注解:

@TableName(autoResultMap = true)

  1. 在字段上面添加注解:

@TableField(value = "list", typeHandler = UserHandler.class) private List<User> list = new ArrayList<>();

二 创建 UserHandler 类

package com.demo.handler;import com.baomidou.mybatisplus.extension.handlers.AbstractJsonTypeHandler;
import com.demo.domain.pojo.User;
import org.apache.ibatis.type.MappedTypes;import java.util.List;@MappedTypes({List.class, User.class})
public class UserHandler extends AbstractJsonTypeHandler<List<User>> {@Overrideprotected List<User> parse(String json) {return JsonUtils.fromString(json, List.class, User.class);}@Overrideprotected String toJson(List<User> obj) {return JsonUtils.objectToString(obj);}
}

如果在使用改类时,只需将文件中的 User 替换为自己的类就行

概括下,这个解析器中的 parse 方法类似咱们的 set 方法,是将数据库中的 json 类型数据转为 List 类型数据;反之,toJson 是将 List 类型数据转为 json 数据格式,最后通过 mybatisplus 存储到数据库中. 具体实现可以看下面的 JsonUtils 工具类.

package com.demo.handler;import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.JavaType;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import org.apache.commons.lang3.StringUtils;import java.io.IOException;public class JsonUtils {private static final ObjectMapper objectMapper = new ObjectMapper();//初始化相关的配置static {//只引用不为空的值objectMapper.setSerializationInclusion(JsonInclude.Include.NON_EMPTY);//取消默认转换timestempobjectMapper.configure(SerializationFeature.WRITE_DURATIONS_AS_TIMESTAMPS, false);//忽略空bean转换错误objectMapper.configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false);//忽略在json中存在,在java对象不存在的错误objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);// 解决jackson2无法反序列化LocalDateTime的问题objectMapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);objectMapper.registerModule(new JavaTimeModule());}/*** 将java对象转换成json字符串** @param obj java 对象* @param <T>* @return*/public static <T> String objectToString(T obj) {if (obj == null) {return null;}try {return obj instanceof String ? (String) obj : objectMapper.writeValueAsString(obj);} catch (JsonProcessingException e) {e.printStackTrace();return null;}}/*** 将json字符串转换成java对象** @param json   字符串* @param tClass 要转换的对象* @param <T>* @return*/public static <T> T getObjetFormString(String json, Class<T> tClass) {if (StringUtils.isBlank(json) || tClass == null) {return null;}try {return tClass.equals(String.class) ? (T) json : objectMapper.readValue(json, tClass);} catch (IOException e) {e.printStackTrace();return null;}}/*** 将字符串转换成java对象** @param json           字符串* @param tTypeReference 对象* @param <T>* @return*/public static <T> T fromString(String json, TypeReference<T> tTypeReference) {if (StringUtils.isBlank(json) || tTypeReference == null) {return null;}try {return tTypeReference.getType().equals(String.class) ? (T) json : objectMapper.readValue(json, tTypeReference);} catch (IOException e) {e.printStackTrace();return null;}}/*** 将json字符串转换成java集合对象** @param json            字符串* @param collectionClass 集合类型* @param elementClazzes  成员类型* @param <T>* @return*/public static <T> T fromString(String json, Class<?> collectionClass, Class<?>... elementClazzes) {JavaType javaType = objectMapper.getTypeFactory().constructParametricType(collectionClass, elementClazzes);try {return objectMapper.readValue(json, javaType);} catch (IOException e) {e.printStackTrace();return null;}}
}

该文件是工具类,配合 UserHandler 类,无需任何改动,直接粘贴即可

三 数据库字段设置

这里只需要将对应的字段类型设置为 json 即可,这里就不贴图了.

还需要将数据库连接后面加上stringtype=unspecified,例如:

jdbc:postgresql://*.*.*.*:5432/demo?currentSchema=demo,SYS_CATALOG&stringtype=unspecified

否则报错:PSQLException: ERROR: column “list” is of type json but expression is of type varchar

额外说明一个事情,可以说是一个 bug,上述效果在 selectById 查询时是成功的,但是在查询返回体是 List 时,list 字段会返回空集合,问了下 ai,说是数据库为了简洁高效,会跳过解析器,断点测试也确实是这样,至于如何解决,我现在是循环单个查询,最后合并,实属无奈,如果有哪位兄弟能搞定,还麻烦大致说下原因.

文章仓促就之,是在已经已经实现的代码上直接说明,如有读者发现文章有不足或是无法实现效果,可以留言.

如果文章确实有效,还望读者点赞收藏,感谢~

http://www.lryc.cn/news/472886.html

相关文章:

  • 公共命名空间,2024年10月的笔记
  • frida脚本,自动化寻址JNI方法
  • ‌MySQL中‌between and的基本用法‌
  • Ceph 存储系统全解
  • C# ftp帮助类 项目实战优化版
  • 栈和队列相关|有效的括号|用队列实现栈|用栈实现队列|设计循环队列(C)
  • 云原生后端开发教程
  • TortoiseSVN小乌龟下载安装(Windows11)
  • Android adb命令获取设备id
  • Skywalking教程一
  • React中管理state的方式
  • 服务器数据恢复—RAID5阵列中部分成员盘重组RAID5阵列后如何恢复原raid5阵列数据?
  • 【Linux】文件切割排序 cut sort
  • 零售EDI:HornBach EDI 项目案例
  • SpringBoot 集成RabbitMQ 实现钉钉日报定时发送功能
  • 基于java ssm springboot女士电商平台系统源码+文档设计
  • Matlab数字信号处理——基于改进小波变换的图像去噪方法(7种去噪算法)
  • leetcode hot100【LeetCode 70. 爬楼梯】java实现
  • Java异常2
  • 2024熵密杯初始题2
  • echarts属性之title
  • VUE errolog, vue 错误集
  • 驱动开发系列13 - Linux tasklet用法介绍
  • redis实现分布式锁,go实现完整code
  • 解析日期、编码
  • 【Qt】QApplication::restoreOverrideCursor():恢复鼠标光标到原始状态的用法解析
  • 重生之“我打数据结构,真的假的?”--2.单链表(无习题)
  • 【有啥问啥】视频插帧算法技术原理详解
  • Leetcode148,109以及二者的合并 -> Tencent面试算法题 - 无序双向链表转BST
  • 【蓝桥杯选拔赛真题77】python计算小球 第十五届青少年组蓝桥杯python选拔赛真题 算法思维真题解析