自定义mybatis map返回类型

1.需求背景

设定订单表order,要根据订单类型统计订单数据,大致sql如下:

select order_type , count(1)  as order_num from order group by order_type;

Mybatis无法将以上sql以指定key:order_type;value:order_num存入至map中。

而Mybatis默认返回的List<Map<String, Object>>,是以每个字段name作为key,字段的值作为value,放入至Map<String,Object>的List数组。
因此自定义一种可以指定key、value字段的Mybatis插件将非常有用。

2.自定义Mybatis拦截器实现Mybaits Map返回类型

有关于Mybatis拦截器的介绍请参阅Mybatis拦截器

由Mybatis源码可知,返回结果集是ResultSetHandler接口的handleResultSets方法实现的,源码如下:

public interface ResultSetHandler {  
      <E> List<E> handleResultSets(Statement stmt) throws SQLException; 
      void handleOutputParameters(CallableStatement cs) throws SQLException;
}

当我们需要返回Map时,只需要对此方法进行拦截,重新组装返回结果数据;当需要拦截时,执行invocation.proceed()。首先我们定义MapParam.java类,用去标识结果集需要拦截,此外在该类中指定返回结果集Map的key和value名称,以及value返回类型。MapParam代码如下:

public class MapParam extends HashMap {   
      // key名称 
      public static final String KEY_FIELD = "keyField";    
      // value名称 
      public static final String VALUE_FIELD = "valueField";    
      // value值类型 
      public static final String VALUE_CLASS = "valueClass";    
      public MapParam(){    }    
      public MapParam(String keyField, String valueField, String valueClass){        
                this.put(KEY_FIELD, keyField);        
                this.put(VALUE_FIELD, valueField);        
                this.put(VALUE_CLASS, valueClass);    }   
      // value值类型枚举类 
      public enum ValueClass {        
                INTEGER("integer"),        
                BIG_DECIMAL("bigDecimal");        
                private String code;       
                public String getCode() {            
                         return code;        
                }        
                ValueClass(String code){            
                        this.code = code;       
                }    
       }
}

通过类MapParam,我们可以定义key和value的字段值,还可以定义value的值类型;

接下来,我们定义MapInterceptor.java,通过对返回结果集方法handleResultSets拦截,返回需要指定的Map数据。少说废话,直接上代码:

@Intercepts(@Signature(method="handleResultSets", type=ResultSetHandler.class, args={Statement.class}))
public class MapInterceptor implements Interceptor {    
         //日志    private static final Logger logger = LoggerFactory.getLogger(MapInterceptor.class);    
         @Override    
        public Object intercept(Invocation invocation) throws Throwable {        
              // 获取代理目标对象        
             Object target = invocation.getTarget();        
             if (target instanceof DefaultResultSetHandler) { 
             DefaultResultSetHandler resultSetHandler = (DefaultResultSetHandler) target;           
             // 利用反射获取参数对象            
             ParameterHandler parameterHandler = reflect(resultSetHandler);
            Object parameterObj = parameterHandler.getParameterObject();
            // 参数对象为MapParam进入处理逻辑            
            if (parameterObj instanceof MapParam) {
                    MapParam mapParam = (MapParam) parameterObj;               
                   // 获取当前statement                
                   Statement stmt = (Statement) invocation.getArgs()[0];               
                  // 根据maoParam返回处理结果               
                   return handleResultSet(stmt.getResultSet(), mapParam);            
             }       
           }       
 return invocation.proceed();   
 }
@Override
public Object plugin(Object target) {    
          return Plugin.wrap(target, this);
}
@Override
public void setProperties(Properties properties) {
}

private Object handleResultSet(ResultSet resultSet, MapParam mapParam){   
          if (null != resultSet){        
          // 获取key field name        
          String keyFieldName = (String)mapParam.get(MapParam.KEY_FIELD);
          // 获取value field name        
          String valueFieldName = (String)mapParam.get(MapParam.VALUE_FIELD);        
         // 值类型        
         String valueClass = (String) mapParam.get(MapParam.VALUE_CLASS);
          List<Object> resultList = new ArrayList<Object>();        
          Map<Object, Object> map = new HashMap<Object, Object>();        
         try {            
              while (resultSet.next()) {                
              Object key = resultSet.getObject(keyFieldName);                
               Object value ;               
               // 根据值类型转换值               
                if (StringUtils.equals(valueClass, MapParam.ValueClass.INTEGER.getCode())) {                    
                     value = resultSet.getInt(valueFieldName);                
               } else if(StringUtils.equals(valueClass, MapParam.ValueClass.BIG_DECIMAL.getCode())) {                    
                     value = resultSet.getBigDecimal(valueFieldName);                
               } else {                    
                    value = resultSet.getObject(valueFieldName);               
             }               
                   map.put(key, value);            
            }        
           } catch (SQLException e) {            
                   logger.error("map interceptor转换异常,{}", e.getMessage());        
           } finally {            
                   // 关闭result set            
                  closeResultSet(resultSet);        
         }       
          resultList.add(map);        
          return resultList;    
}    
return  null;
}

private void closeResultSet(ResultSet resultSet) {    
         try {       
                 if (resultSet != null) {            
                        resultSet.close();        
                 }    
             } catch (SQLException e) {        
                  logger.error("关闭 result set异常,{}", e.getMessage());   
                 }
}

private ParameterHandler reflect(DefaultResultSetHandler resultSetHandler){
      Field field = ReflectionUtils.findField(DefaultResultSetHandler.class, "parameterHandler");    
      field.setAccessible(true);    
      Object value = null;    
      try {       
               value = field.get(resultSetHandler);    
       } catch (Exception e) {        
              logger.error("默认返回结果集反射参数对象异常,{}", e.getMessage()); 
      }    
     return (ParameterHandler)value;
}

@Intercepts(@Signature(method="handleResultSets", type=ResultSetHandler.class, args={Statement.class}))注解代码含义,@Intercepts用于表示该类为拦截器,@Signature用于标识需要拦截的方法名、返回类型及方法参数值。

进入ResultSetHandler的handleResultSets方法,都会进入此拦截器,当需要的请求参数为类MapParam时,执行处理,否则执行invocation.proceed()
代码写的很详细,再次无需再重复讲述。

3.具体实现

注册拦截器:

<bean id="sqlSessionFactory" 
        <property name="plugins">        
              <array>            
                    <bean class="com.test.common.interceptor.MapInterceptor"/>        
              </array>    
          </property>
</bean>

dao层代码:

MapParam params = new MapParam("orderType","orderNum",MapParam.ValueClass.INTEGER.getCode());
Map<String,Integer> find(MapParam params);

xml代码:

<select id="find" resultType="map" parameterType="MapParam">  
    select order_type as orderType , count(1)  as orderNum from order group by order_type;
</select>  

打印结果格式:
“mobileOrder”:“100”
“partsOrder”:“200”
“normalOrder”:“500”

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,378评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,356评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,702评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,259评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,263评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,036评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,349评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,979评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,469评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,938评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,059评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,703评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,257评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,262评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,485评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,501评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,792评论 2 345

推荐阅读更多精彩内容

  • 1. 简介 1.1 什么是 MyBatis ? MyBatis 是支持定制化 SQL、存储过程以及高级映射的优秀的...
    笨鸟慢飞阅读 5,429评论 0 4
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,599评论 18 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,581评论 18 399
  • 隔了半年之后的苹果。ヽ(  ̄д ̄;)ノ
    Mister_阅读 196评论 0 2