Mybatis返回Map时,如果值为空的话,那么key也不会被加入到map中的解决办法

本文介绍在使用MyBatis框架时如何处理查询结果中为空的字段,通过配置让其返回null而非忽略,并提供自定义类型处理器实现空字符串返回。

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

转自 : http://blog.youkuaiyun.com/renfufei/article/details/39646207



前提:

项目中集成Mybatis与Spring,使用的是Mybatis3.2.7,以及Spring4.0.5,mybatis-spring-1.2.2;

因为项目组成员想要偷懒,将数据从DB中查询出来时需要将字段映射为Map,而不想封装成Bean .

默认情况下, Mybatis对Map的解析生成, 如果值(value)为null的话,那么key也不会被加入到map中 .
于是对Map遍历时,key就遍历不到, 因为前端工具的需要,必须有这个key ,网上搜索后发现需要设置  callSettersOnNulls  这个属性.

那就设置呗, 在 sqlSessionFactory 的定义中,指定 configLocation 属性,指向另一个文件,如下所示



解决方法:

转自 : http://blog.youkuaiyun.com/lulidaitian/article/details/70941769

一、查询sql添加每个字段的判断空

IFNULL(rate,'') as rate
 
  • 1

二、ResultType利用实体返回,不用map

三、springMVC+mybatis查询数据,返回resultType=”map”时,如果数据为空的字段,则该字段省略不显示,可以通过添加配置文件,规定查询数据为空是则返回null。

主要是在mybatis的配置文件中添加下面的配置就可以把空的列封装成"null"返回,然后前台根据需求过滤一下就行了

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration PUBLIC "-//mybatis.org//DTD SQL MAP Config 3.1//EN"
    "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
  <settings>
    <setting name="callSettersOnNulls" value="true"/>
  </settings>
</configuration>
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

spring-mybatis.xml

<!-- spring和MyBatis完美整合,添加mybatis的配置映射文件 -->
  <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
    <property name="dataSource" ref="dataSource"/>
    <property name="configLocation" value="classpath:mybatis-configuration.xml"/>
    <!-- 自动扫描mapping.xml文件 -->
    <property name="mapperLocations" value="classpath:mapping/*.xml"></property>
  </bean>
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

如果想要配置rate的默认值,例如“”字符串,则可以建立一个类,实现Mybatis的TypeHandler接口

public class EmptyStringIfNull implements TypeHandler<String> {

    @Override
    public String getResult(ResultSet rs, String columnName) throws SQLException {
     return (rs.getString(columnName) == null) ? "" : rs.getString(columnName); 
    }

    @Override
    public String getResult(ResultSet rs, int columnIndex) throws SQLException {
     return (rs.getString(columnIndex) == null) ? "" : rs.getString(columnIndex);
    }
    @Override
    public String getResult(CallableStatement cs, int columnIndex)   throws SQLException {
     return (cs.getString(columnIndex) == null) ? "" : cs.getString(columnIndex);
    }
    @Override
    public void setParameter(PreparedStatement ps, int arg1, String str, JdbcType jdbcType) throws SQLException { }}
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17

在sql.xml文件定义与使用如下如下

<resultMap id="find" type="java.util.LinkedHashMap">
    <result property="name" column="name" />
    <result property="phone" column="phone" />
    <result property="rate" column="rate" typeHandler="com.mybatis.EmptyStringIfNull"/>
</resultMap>

### 将 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、付费专栏及课程。

余额充值