【Mybatis-Plus】PostgreSQL之JSONB自动映射对象

开发TypeHandler

如果数据库字段是字符类型,可以直接使用Mybatis-Plus内置的JacksonTypeHandler,如果是像PostgreSQL数据库的JSONB类型,则需要自定义开发一个TypeHandler,可以继承JacksonTypeHandler,然后覆写父类的方法,代码如下:

java 复制代码
/**
 * PostgreSQL数据库中的JSON、JSONB字段类型的映射
 *
 * @author tangheng
 */
@MappedTypes({Object.class})
@MappedJdbcTypes(JdbcType.OTHER)//JSONB对应JdbcType.OTHER
public class JsonbTypeHandler extends JacksonTypeHandler {

    private static final PGobject jsonObject = new PGobject();
    private static final String JSONB = "jsonb";
    private static final String JSON = "json";

    public JsonbTypeHandler(Class<?> type) {
        super(type);
    }

    /**写数据库时,把java对象转成JSONB类型*/
    @Override
    public void setNonNullParameter(PreparedStatement ps, int i, Object parameter, JdbcType jdbcType) throws SQLException {
        if (ps != null) {
            jsonObject.setType(JSONB);
            jsonObject.setValue(toJson(parameter));
            ps.setObject(i, jsonObject);
        }
    }

    /**读数据时,把JSONB类型的字段转成java对象*/
    @Override
    public Object getNullableResult(ResultSet rs, String columnName) throws SQLException {
        Object v = rs.getObject(columnName);
        return convertDbToJavaObject(v);
    }

    /**读数据时,把JSONB类型的字段转成java对象*/
    @Override
    public Object getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
        Object v = rs.getObject(columnIndex);
        return convertDbToJavaObject(v);
    }

    /**读数据时,把JSONB类型的字段转成java对象*/
    @Override
    public Object getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
        Object v = cs.getObject(columnIndex);
        return convertDbToJavaObject(v);
    }

    /**读数据时,把JSONB类型的字段转成java对象*/
    private Object convertDbToJavaObject(Object v) {
        if(Objects.isNull(v)) {
            return v;
        }
        if (!PGobject.class.isAssignableFrom(v.getClass())) {
            return v;
        }
        PGobject p = (PGobject) v;
        if(!StringUtils.equalsAnyIgnoreCase(p.getType(), JSONB, JSON)) {
            return v;
        }
        String pv = p.getValue();
        if(StringUtils.isBlank(pv)) {
            return v;
        }

        return parse(p.getValue());
    }
}

使用TypeHandler

  • 在库表实体类上加的@TableName的注解,设置autoResultMap属性
  • @TableField注解指定typeHandler
java 复制代码
@TableName(value = "t_user", autoResultMap = true)
public class UserPo {
    
    @TableField(value = "user_jobs", typeHandler = JsonbTypeHandler.class)
    private List<UserJob> userJobs;

    @NoArgsConstructor
    @AllArgsConstructor
    @Data
    public static class UserJob {
        private String id;
        private String name;
    }
}

总结

  • 使用JacksonTypeHandler可以帮开发者自动处理json格式与java对象之间的映射关系,不需要每次手动做序列化和反序列化的工作了。
  • 如果使用的数据库有自己特有的字段类型,也可以自定义开发一个TypeHandler,达到自动转换的目的。
相关推荐
Java小白笔记9 小时前
关于使用Mybatis-Plus 自动填充功能失效问题
spring boot·后端·mybatis
亚林瓜子12 小时前
Jackson注解屏蔽某些字段读取权限
spring·json·jackson
计算机学姐12 小时前
基于SpringBoot+Vue的篮球馆会员信息管理系统
java·vue.js·spring boot·后端·mysql·spring·mybatis
程序员大金13 小时前
基于SpringBoot+Vue+MySQL的智能物流管理系统
java·javascript·vue.js·spring boot·后端·mysql·mybatis
惜.己17 小时前
MyBatis中一对多关系的两种处理方法
java·开发语言·后端·sql·mysql·mybatis·idea
终末圆17 小时前
MyBatis动态SQL中的`if`标签使用【后端 19】
java·数据结构·数据库·sql·算法·spring·mybatis
不惑_18 小时前
最佳实践 · 如何高效索引MySQL JSON字段
java·mysql·json
飞翔的佩奇19 小时前
Java项目: 基于SpringBoot+mybatis+maven医院管理系统(含源码+数据库+任务书+开题报告+毕业论文)
java·数据库·spring boot·毕业设计·maven·mybatis·医院管理系统
Z_W_H_20 小时前
【PostgreSQL】安装及使用(Navicat/Arcgis),连接(C#)
数据库·postgresql
这孩子叫逆20 小时前
35. MyBatis中的缓存失效机制是如何工作的?
java·spring·mybatis