MyBatis-Plus 字段类型处理器TypeHandler

字段类型处理器

类型处理器,用于 JavaType 与 JdbcType 之间的转换,用于 PreparedStatement 设置参数值和从 ResultSet 或 CallableStatement 中取出一个值,本文讲解 mybaits-plus 内置常用类型处理器如何通过TableField注解快速注入到 mybatis 容器中。

  • JSON 字段类型
@Data
@Accessors(chain = true)
@TableName(autoResultMap = true)
public class User {
    private Long id;

    ...


    /**
     * 注意!! 必须开启映射注解
     *
     * @TableName(autoResultMap = true)
     *
     * 以下两种类型处理器,二选一 也可以同时存在
     *
     * 注意!!选择对应的 JSON 处理器也必须存在对应 JSON 解析依赖包
     */
    @TableField(typeHandler = JacksonTypeHandler.class)
    // @TableField(typeHandler = FastjsonTypeHandler.class)
    private OtherInfo otherInfo;

}

该注解对应了 XML 中写法为

<result column="other_info" jdbcType="VARCHAR" property="otherInfo" typeHandler="com.baomidou.mybatisplus.extension.handlers.JacksonTypeHandler" />

 

实际使用

1,准备工作

(1)MyBatis 中的 TypeHandler 类型处理器用于 JavaType 与 JdbcType 之间的转换,假设我们用户表中有一个联系方式字段,类型为字符串:

原文:SpringBoot - MyBatis-Plus使用详解16(字段类型处理器TypeHandler)

 

(2)而对应的实体类代码如下,可以看到实体类中 contact 属性类型为 Map。由于与数据库字段类型不匹配,如果不做任何处理的话无论是查询还是插入都会报错。

    这里我们通过 @TableField 注解将 FastjsonTypeHandler 这个类型处理器快速注入到 mybatis 容器中:

注意:使用字段类型处理器时,必须开启映射注解 @TableName(autoResultMap = true)。否则插入没问题,但查询时该字段会为空。

1

2

3

4

5

6

7

8

9

10

@Data

@TableName(autoResultMap = true)

public class UserInfo {

    private Integer id;

    private String userName;

    private String passWord;

    private Integer age;

    @TableField(typeHandler = FastjsonTypeHandler.class)

    private Map<String, String> contact;

}


(3)由于这里用到了 Fastjson 这个 JSON 处理器,所以项目中也要添加相关的依赖:

1

2

3

4

5

<dependency>

    <groupId>com.alibaba</groupId>

    <artifactId>fastjson</artifactId>

    <version>1.2.56</version>

</dependency>

 

2,开始测试

(1)首先创建一个对象插入到数据库,可以看到虽然对象里的 contact 是一个 Map,但会自动转成 JSON 字符串存到数据库中:

1

2

3

4

5

6

7

8

9

UserInfo user = new UserInfo();

user.setUserName("hangge");

user.setPassWord("123");

//添加联系方式

Map<String, String> contact = new HashMap<>();

contact.put("phone""010-1234567");

contact.put("tel""13388889999");

user.setContact(contact);

userInfoMapper.insert(user);

原文:SpringBoot - MyBatis-Plus使用详解16(字段类型处理器TypeHandler)


(2)接着测试下查询,可以发现数据库的 JSON 字符串也会自动转换成 Map:

1

2

List<UserInfo> users = userInfoMapper.selectList(null);

System.out.println(users);

原文:SpringBoot - MyBatis-Plus使用详解16(字段类型处理器TypeHandler)

 

附:自定义 TypeHandler 类型处理器

1,准备工作

(1)如果我们需要实现一个自定义的 TypeHandler 类型处理器,则需继承 ListTypeHandler 接口,接口作用是用于指定 jdbc 与 java 的数据类型间对应关系处理。接口代码如下:

1

2

3

4

5

6

7

8

public interface TypeHandler<T> {

  // 保存操作,数据入库之前时数据处理

  void setParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException;

  //下面三个则是,从数据库加载数据后,vo对象封装前的数据处理

  T getResult(ResultSet rs, String columnName) throws SQLException;

  T getResult(ResultSet rs, int columnIndex) throws SQLException;

  T getResult(CallableStatement cs, int columnIndex) throws SQLException;

}


(2)这里我们定义一个 list 类型转 varchar(逗号隔开的字符串)的类型处理器,即 list [1,2,3] ==》varchar 1,2,3

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

@MappedJdbcTypes(JdbcType.VARCHAR)  //数据库类型

@MappedTypes({List.class})          //java数据类型

@Log

public class ListTypeHandler implements TypeHandler<List<String>> {

 

    @Override

    public void setParameter(PreparedStatement ps, int i,

                             List<String> parameter, JdbcType jdbcType) throws SQLException {

        log.info("method ====>>> setParameter");

        String hobbys = dealListToOneStr(parameter);

        ps.setString(i , hobbys);

    }

 

    /**

     * 集合拼接字符串

     * @param parameter

     * @return

     */

    private String dealListToOneStr(List<String> parameter){

        if(parameter == null || parameter.size() <=0)

            return null;

        String res = "";

        for (int i = 0 ;i < parameter.size(); i++) {

            if(i == parameter.size()-1){

                res+=parameter.get(i);

                return res;

            }

            res+=parameter.get(i)+",";

        }

        return null;

    }

     

    @Override

    public List<String> getResult(ResultSet rs, String columnName)

            throws SQLException {

        log.info("method ====>>> getResult(ResultSet rs, String columnName)");

        return Arrays.asList(rs.getString(columnName).split(","));

    }

 

    @Override

    public List<String> getResult(ResultSet rs, int columnIndex)

            throws SQLException {

        log.info("method ====>>> getResult(ResultSet rs, int columnIndex)");

        return Arrays.asList(rs.getString(columnIndex).split(","));

    }

 

    @Override

    public List<String> getResult(CallableStatement cs, int columnIndex) throws SQLException{

        log.info("method ====>>> getResult(CallableStatement cs, int columnIndex)");

        String hobbys = cs.getString(columnIndex);

        return Arrays.asList(hobbys.split(","));

    }

}


(3)接着在对应的实例类中使用 @TableField 注解指定使用这个自定义的字段类型处理器:

1

2

3

4

5

6

7

8

9

10

@Data

@TableName(autoResultMap = true)

public class UserInfo {

    private Integer id;

    private String userName;

    private String passWord;

    private Integer age;

    @TableField(typeHandler = ListTypeHandler.class)

    private List<String> contact;

}

 

2,开始测试

(1)首先创建一个对象插入到数据库,可以看到虽然对象里的 contact 是一个 List,但会自动转成逗号隔开的字符串存到数据库中:

1

2

3

4

5

6

7

8

9

10

UserInfo user = new UserInfo();

user.setUserName("hangge");

user.setPassWord("123");

//添加联系方式

List<String> contact = new ArrayList<>();

contact.add("010-1234567");

contact.add("13388889999");

contact.add("10001");

user.setContact(contact);

userInfoMapper.insert(user);

原文:SpringBoot - MyBatis-Plus使用详解16(字段类型处理器TypeHandler)


(2)接着测试下查询,可以发现数据库的字符串也会自动还原成 List:

1

2

List<UserInfo> users = userInfoMapper.selectList(null);

System.out.println(users);

原文:SpringBoot - MyBatis-Plus使用详解16(字段类型处理器TypeHandler)


原文出自:www.hangge.com  转载请保留原文链接:https://www.hangge.com/blog/cache/detail_2926.html

  • 8
    点赞
  • 31
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
Mybatis-plus支持读取json类型数据,方法如下: 1. 首先,在数据库中创建一个json类型字段,如下所示: CREATE TABLE `user` ( `id` int(11) NOT NULL AUTO_INCREMENT, `name` varchar(255) DEFAULT NULL, `address` varchar(255) DEFAULT NULL, `info` json DEFAULT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8; 2. 在Java实体类中,为info字段添加注解@JSONField,如下所示: public class User { private Long id; private String name; private String address; @JSONField(serializeUsing = JSONObjectSerializer.class, deserializeUsing = JSONObjectDeserializer.class) private JSONObject info; // 省略getter和setter方法 } 3. 在Mybatis-plus的mapper接口中,使用@Results和@Result注解映射json字段,如下所示: @Results(id = "userResultMap", value = { @Result(property = "id", column = "id", id = true), @Result(property = "name", column = "name"), @Result(property = "address", column = "address"), @Result(property = "info", column = "info", javaType = JSONObject.class, typeHandler = JSONTypeHandler.class), }) public interface UserMapper extends BaseMapper<User> {} 4. 在配置文件中,配置typeHandlersPackage为org.apache.ibatis.type,如下所示: mybatis-plus: configuration: map-underscore-to-camel-case: true log-impl: org.apache.ibatis.logging.stdout.StdOutImpl type-handlers-package: org.apache.ibatis.type 5. 最后,就可以在mapper接口中使用Mybatis-plus提供的方法读取json类型数据了,如下所示: User user = userMapper.selectOne(Wrappers.<User>lambdaQuery().eq(User::getId, 1L)); JSONObject info = user.getInfo(); 以上就是Mybatis-plus读取json类型数据的方法实现。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值