mybatis返回Map结果集value为Null的情况丢失key解决办法

本文介绍了一种在使用MyBatis框架时遇到的问题:当查询结果中某个字段值为Null时,返回的Map结果集中对应字段的key会丢失。通过在DataSourceConfig.java文件中设置config.setCallSettersOnNulls(true),可以确保即使字段值为Null也会返回相应的字段名称。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

mybatis返回Map结果集value为Null的情况下丢失key的解决办法

本博主之前一直是网络资源的索取者,本着开源精神愿意把自己在开发过程中遇到过的问题以及解决方案分享给大家,这是我的第一篇博客,希望以后能坚持写博客,让大家少走弯路、少踩坑。
废话少说,直入主题!
springcloud微服务集成mybatis动态多数据源在网上有很多文章可以搜到,这里就不重复造轮子了。
如题,网上的很多解决方案比如:
尝试在application.yml 文件中配置 mybatis:callSettersOnNulls:true,对于springboot启动加载yml的单一数据源是有效的,但是在动态多数据源的情况下,是需要自己手动增加数据源配置类:DataSourceConfig.java,并且需要在启动类加上了@SpringBootApplication(exclude ={DataSourceAutoConfiguration.class})注解。
因此,上述解决方案就没用了,下面给出我的解决方案,亲测有效:
1.xml文件

<mapper namespace="com.jess.order.dao.extend.VersionExtendMapper" >
  <resultMap id="baseResultMap" type="java.util.LinkedHashMap">
    <result property="id" column="id" />
    <result property="name" column="name" />
    <result property="versionNo" column="version_no" />
    <result property="createtime" column="create_time" />
    <result property="updatetime" column="update_time" />
  </resultMap>

  <select id="find" resultMap="baseResultMap" >
    select * from com_version
  </select>

如上,返回的map格式如下:
“data”: [
{
“id”: 1,
“name”: “版本名称1”,
“versionNo”: “版本号1”,
“createtime”: “2018-09-08 10:22:24”,
“updatetime”: “2018-09-08 10:33:54”
},
{
“id”: 2,
“name”: “版本名称2”,
“versionNo”: “版本号2”,
“createtime”: “2018-09-08 10:22:24”,
“updatetime”: “2018-09-08 10:33:54”
}]
如果数据库对应的id=1的字段version_no的值是Null,那么就会出现下面的情况:
“data”: [
{
“id”: 1,
“name”: “版本名称1”,
“createtime”: “2018-09-08 10:22:24”,
“updatetime”: “2018-09-08 10:33:54”
},
{
“id”: 2,
“name”: “版本名称2”,
“versionNo”: “版本号2”,
“createtime”: “2018-09-08 10:22:24”,
“updatetime”: “2018-09-08 10:33:54”
}]
也就是说Map中的key丢失了,在前后端分离的项目中,前端获取丢失的字段直接报错!
这个问题困扰了我好几天,阅读源码后总算是解决了问题,方案如下:
修改DataSourceConfig.java文件

@Bean
    public SqlSessionFactory sqlSessionFactory(DataSource dynamicDataSource) throws Exception {
        SqlSessionFactoryBean bean = new SqlSessionFactoryBean();
        bean.setDataSource(dynamicDataSource);
        bean.setMapperLocations(new PathMatchingResourcePatternResolver().getResources("classpath*:mapper/**/**.xml"));
        **org.apache.ibatis.session.Configuration config = bean.getObject().getConfiguration();
        config.setCallSettersOnNulls(true);
        bean.setConfiguration(config);**
        return bean.getObject();
    }

上述代码加粗的字体中:config.setCallSettersOnNulls(true);表示设置结果为Null也返回相应的字段名称。就这么简单,解决了!so easy

### 将 MyBatis 返回Map 中的 Key 转为大写 在 MyBatis 中,默认返回的 `Map` 类型结果是基于数据库查询字段名作为键的,这些键通常是小写的(具体取决于数据库的字段命名规则)。如果需要将返回的 `Map` 的键转为大写,可以通过自定义结果处理器来实现。以下是实现方法: #### 方法:使用 `ResultSetHandler` 自定义处理逻辑 MyBatis 提供了对 `ResultSetHandler` 的扩展支持,允许开发者自定义结果的处理逻辑。通过实现 `org.apache.ibatis.executor.resultset.DefaultResultSetHandler` 或直接在 Mapper 方法中使用 `@ResultMap` 和 `@MapKey` 注解,可以实现对返回结果的修改。 以下是一个完整的代码示例: ```java import org.apache.ibatis.executor.resultset.DefaultResultSetHandler; import org.apache.ibatis.session.SqlSession; import java.sql.ResultSet; import java.sql.ResultSetMetaData; import java.util.HashMap; import java.util.Map; public class UppercaseKeyResultSetHandler extends DefaultResultSetHandler { @Override protected Map<String, Object> handleRow(ResultSet rs) throws Exception { Map<String, Object> resultMap = new HashMap<>(); ResultSetMetaData metaData = rs.getMetaData(); int columnCount = metaData.getColumnCount(); for (int i = 1; i <= columnCount; i++) { String columnName = metaData.getColumnName(i).toUpperCase(); // 将列名转为大写 Object value = rs.getObject(i); resultMap.put(columnName, value); } return resultMap; } public static void main(String[] args) { try (SqlSession session = MyBatisUtil.getSqlSessionFactory().openSession()) { UppercaseKeyResultSetHandler handler = new UppercaseKeyResultSetHandler(); // 使用自定义的 ResultSetHandler 处理查询结果 session.select("yourMapperId", null, handler); } } } ``` 上述代码通过继承 `DefaultResultSetHandler` 并重写 `handleRow` 方法,实现了对每一行数据的键转换逻辑[^4]。 #### 方法:使用拦截器(Interceptor) 另一种方式是通过 MyBatis 的拦截器机制,在结果返回之前对 `Map` 的键进行修改。以下是一个拦截器的实现示例: ```java import org.apache.ibatis.executor.resultset.DefaultResultSetHandler; import org.apache.ibatis.plugin.*; import java.sql.ResultSet; import java.util.Map; import java.util.Properties; @Intercepts({ @Signature(type = DefaultResultSetHandler.class, method = "handleResultSets", args = {ResultSet.class}) }) public class UppercaseKeyInterceptor implements Interceptor { @Override public Object intercept(Invocation invocation) throws Throwable { Object result = invocation.proceed(); if (result instanceof Map) { Map<String, Object> resultMap = (Map<String, Object>) result; Map<String, Object> upperCaseMap = new HashMap<>(); resultMap.forEach((key, value) -> upperCaseMap.put(key.toUpperCase(), value)); return upperCaseMap; } return result; } @Override public Object plugin(Object target) { return Plugin.wrap(target, this); } @Override public void setProperties(Properties properties) { } } ``` 在 MyBatis 配置文件中注册该拦截器即可生效: ```xml <plugins> <plugin interceptor="com.example.UppercaseKeyInterceptor"/> </plugins> ``` #### 注意事项 - 如果查询结果包含嵌套结构(如 `List<Map>`),需要递归处理每个 `Map`。 - 拦截器方式适用于全局场景,而自定义 `ResultSetHandler` 更适合特定查询的需求[^5]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值