溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

MyBatis中@MapKey怎么用

發(fā)布時間:2022-02-08 09:45:02 來源:億速云 閱讀:885 作者:小新 欄目:開發(fā)技術(shù)

這篇文章將為大家詳細講解有關(guān)MyBatis中@MapKey怎么用,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

MyBatis @MapKey的妙用

背景

在實際開發(fā)中,有一些場景需要我們返回主鍵或者唯一鍵為Key、Entity為Value的Map集合,如Map<Long, User>,之后我們就可以直接通過map.get(key)的方式來獲取Entity。

實現(xiàn)

MyBatis為我們提供了這種實現(xiàn),Dao示例如下:

public interface UserDao {    
    @MapKey("id")
    Map<Long, User> selectByIdList(@Param("idList") List<Long> idList);    
}

需要注意的是:如果Mapper.xml中的select返回類型是List的元素,上面示例的話,resultType是User,因為selectMap查詢首先是selectList,之后才是處理List。

源碼分析

package org.apache.ibatis.session.defaults;
public class DefaultSqlSession implements SqlSession {
  ... ...
  public <K, V> Map<K, V> selectMap(String statement, Object parameter, String mapKey, RowBounds rowBounds) {
    final List<?> list = selectList(statement, parameter, rowBounds);
    final DefaultMapResultHandler<K, V> mapResultHandler = new DefaultMapResultHandler<K, V>(mapKey,
        configuration.getObjectFactory(), configuration.getObjectWrapperFactory());
    final DefaultResultContext context = new DefaultResultContext();
    for (Object o : list) {
      context.nextResultObject(o);
      mapResultHandler.handleResult(context);
    }
    Map<K, V> selectedMap = mapResultHandler.getMappedResults();
    return selectedMap;
  }  
  ... ...
}

selectMap方法其實是在selectList后的進一步處理,通過mapKey獲取DefaultMapResultHandler類型的結(jié)果處理器,然后遍歷list,調(diào)用handler的handleResult把每個結(jié)果處理后放到map中,最后返回map。

package org.apache.ibatis.executor.result;
public class DefaultMapResultHandler<K, V> implements ResultHandler {
  private final Map<K, V> mappedResults;  
  ... ...
  public void handleResult(ResultContext context) {
    // TODO is that assignment always true?
    final V value = (V) context.getResultObject();
    final MetaObject mo = MetaObject.forObject(value, objectFactory, objectWrapperFactory);
    // TODO is that assignment always true?
    final K key = (K) mo.getValue(mapKey);
    mappedResults.put(key, value);
  }
  ... ...  
}

可以看出DefaultMapResultHandler是通過mapKey從元數(shù)據(jù)中獲取K,然后mappedResults.put(key, value)放到map中。

思考

@MapKey這種處理是在查詢完后做的處理,實際上我們也可以自己寫邏輯將List轉(zhuǎn)成Map,一個Lambda表達式搞定,如下:

  List<User> list = userDao.selectByIdList(Arrays.asList(1,2,3));
  Map<Integer, User> map = list.stream().collect(Collectors.toMap(User::getId, user -> user));

Mybatis @MapKey分析

先上例子

 @Test
  public void testShouSelectStudentUsingMapperClass(){
    //waring 就使用他作為第一個測試類
    try (SqlSession session = sqlMapper.openSession()) {
      StudentMapper mapper = session.getMapper(StudentMapper.class);
      System.out.println(mapper.listStudentByIds(new int[]{1,2,3}));
    }
  }
  @MapKey("id")
  Map<Integer,StudentDo> listStudentByIds(int[] ids);
  <select id="listStudentByIds" resultType="java.util.Map">
    select  * from t_student
    where  id in
      <foreach collection="array" open="(" separator="," close=")" item="item">
            #{item}
      </foreach>
  </select>

結(jié)果

MyBatis中@MapKey怎么用

1. MapKey注解有啥功能

Mapkey可以讓查詢的結(jié)果組裝成Map,Map的key是@MapKey指定的字段,Value是實體類。如上圖所示

2. MapKey的源碼分析

還是從源碼分析一下他是怎么實現(xiàn)的,要注意MapKey不是寫在Xml中的,而是標注在方法上的。所以,對于XML文件來說,他肯定不是在解析文件的時候操作的。對于Mapper注解實現(xiàn)來說,理論上來說是在解析的時候用的,但是對比XML的解析來說,應(yīng)該不是。多說一點,想想Spring ,開始的時候都是XML,最后采用的注解,并且注解的功能和XML的對應(yīng)起來,所以在解析XML是怎么解析的,在解析注解的時候就應(yīng)該是怎么解析的。

還是老套路,點點看看,看看哪里引用到了他,從下圖看到,用到的地方一個是MapperMethod,一個是MapperAnnotationBuilder,在MapperAnnotationBuilder里面只是判斷了一下,沒有啥實質(zhì)性的操作,這里就不用管。只看前者。

MyBatis中@MapKey怎么用

1. MapperMethod對MapKey的操作

看過之前文章的肯定知道,MapperMethod是在哪里創(chuàng)建的。這個是在調(diào)用mapper接口的查詢的時候創(chuàng)建的。接口方法的執(zhí)行最終會調(diào)用到這個對象的execute方法

MapperMethod里面包含兩個對象SqlCommand和MethodSignature,就是在MethodSignature里面引用了MapKey的

 public MethodSignature(Configuration configuration, Class<?> mapperInterface, Method method) {
     //判斷此方法的返回值的類型
      Type resolvedReturnType = TypeParameterResolver.resolveReturnType(method, mapperInterface);
      if (resolvedReturnType instanceof Class<?>) {
        this.returnType = (Class<?>) resolvedReturnType;
      } else if (resolvedReturnType instanceof ParameterizedType) {
        this.returnType = (Class<?>) ((ParameterizedType) resolvedReturnType).getRawType();
      } else {
        this.returnType = method.getReturnType();
      }
     // 返回值是否為空
      this.returnsVoid = void.class.equals(this.returnType);
     // 返回是否是一個列表或者數(shù)組
      this.returnsMany = configuration.getObjectFactory().isCollection(this.returnType) || this.returnType.isArray();
      // 返回值是否返回一個游標
      this.returnsCursor = Cursor.class.equals(this.returnType);
   
    // 返回值是否是一個Optional
      this.returnsOptional = Optional.class.equals(this.returnType);
   
     // 重點來了,這里會判斷返回值是一個MapKey,并且會將MapKey里Value的值賦值給MapKey
      this.mapKey = getMapKey(method);
   
     // 返回值是否是一個map
      this.returnsMap = this.mapKey != null;
      this.rowBoundsIndex = getUniqueParamIndex(method, RowBounds.class);
      this.resultHandlerIndex = getUniqueParamIndex(method, ResultHandler.class); //找到方法參數(shù)里面 第一個 參數(shù)類型為ResultHandler的值
      // 這里是處理方法參數(shù)里面的param注解, 注意方法參數(shù)里面有兩個特殊的參數(shù) RowBounds和 ResultHandler
      // 這里會判斷@param指定的參數(shù),并且會將這些參數(shù)組成一個map,key是下標,value是param指定的參數(shù),如果沒有,就使用方法參數(shù)名
      this.paramNameResolver = new ParamNameResolver(configuration, method);
    }

上面的代碼這次最重要的是mapKey的賦值操作getMapKey,來看看他是什么樣子

private String getMapKey(Method method) {
      String mapKey = null;
      if (Map.class.isAssignableFrom(method.getReturnType())) {
        final MapKey mapKeyAnnotation = method.getAnnotation(MapKey.class);
        if (mapKeyAnnotation != null) {
          mapKey = mapKeyAnnotation.value();
        }
      }
      return mapKey;
    }

上面介紹了MapKey是在哪里解析的,下面分析Mapkey是怎么應(yīng)用的,拋開所有的不說,圍繞查詢來說。經(jīng)過上面的介紹。已經(jīng)對查詢的流程很清晰了,因為查詢還是普通的查詢,所以,MapKey在組裝值的時候才會發(fā)送作用,下面就看看吧

還是老套路,既然賦值給MethodSignature的mapKey了,點點看看,哪里引用了他

MyBatis中@MapKey怎么用

下面的沒有啥可看的,看看上面,在MapperMethod里面用到了,那就看看

//看這個名字就能知道,這是一個執(zhí)行Map查詢的操作
private <K, V> Map<K, V> executeForMap(SqlSession sqlSession, Object[] args) {
    Map<K, V> result;
    Object param = method.convertArgsToSqlCommandParam(args);
    if (method.hasRowBounds()) {
      RowBounds rowBounds = method.extractRowBounds(args);
      // 將Map傳遞給sqlSession了,那就一直往下走
      result = sqlSession.selectMap(command.getName(), param, method.getMapKey(), rowBounds);
    } else {
      result = sqlSession.selectMap(command.getName(), param, method.getMapKey());
    }
    return result;
  }

一直點下去,就看到下面的這個了,可以看到,這里將mapKey傳遞給了DefaultMapResultHandler,對查詢的結(jié)果進行處理。

  @Override
  public <K, V> Map<K, V> selectMap(String statement, Object parameter, String mapKey, RowBounds rowBounds) {
     //這已經(jīng)做了查詢了
    final List<? extends V> list = selectList(statement, parameter, rowBounds);
    final DefaultMapResultHandler<K, V> mapResultHandler = new DefaultMapResultHandler<>(mapKey,
            configuration.getObjectFactory(), configuration.getObjectWrapperFactory(), configuration.getReflectorFactory());
    final DefaultResultContext<V> context = new DefaultResultContext<>();
    // 遍歷list,利用mapResultHandler處理List
    for (V o : list) {
      context.nextResultObject(o);
      mapResultHandler.handleResult(context);
    }
    return mapResultHandler.getMappedResults();
  }

這里很明確了,先做正常的查詢,在對查詢到的結(jié)果做處理(DefaultMapResultHandler)。

2. DefaultMapResultHandler是什么
/**
 * @author Clinton Begin
 */
public class DefaultMapResultHandler<K, V> implements ResultHandler<V> {
  private final Map<K, V> mappedResults;
  private final String mapKey;
  private final ObjectFactory objectFactory;
  private final ObjectWrapperFactory objectWrapperFactory;
  private final ReflectorFactory reflectorFactory;
  @SuppressWarnings("unchecked")
  public DefaultMapResultHandler(String mapKey, ObjectFactory objectFactory, ObjectWrapperFactory objectWrapperFactory, ReflectorFactory reflectorFactory) {
    this.objectFactory = objectFactory;
    this.objectWrapperFactory = objectWrapperFactory;
    this.reflectorFactory = reflectorFactory;
    this.mappedResults = objectFactory.create(Map.class);
    this.mapKey = mapKey;
  }
 // 邏輯就是這里,
  @Override
  public void handleResult(ResultContext<? extends V> context) {
    //拿到遍歷的list的當(dāng)前值。
    final V value = context.getResultObject();
    //構(gòu)建MetaObject,
    final MetaObject mo = MetaObject.forObject(value, objectFactory, objectWrapperFactory, reflectorFactory);
    // TODO is that assignment always true?
    // 獲取mapKey指定的屬性,放在mappedResults里面。
    final K key = (K) mo.getValue(mapKey);
    mappedResults.put(key, value);
  }
 // 返回結(jié)果
  public Map<K, V> getMappedResults() {
    return mappedResults;
  }
}

這里的邏輯很清晰,對查詢查到的list。做遍歷,利用反射獲取MapKey指定的字段,并且組成Map,放在一個Map(mappedResults,這默認就是HashMap)里面。

問題?

1, 從結(jié)果中獲取mapkey字段的操作,這個字段總是有的嗎?

MyBatis中@MapKey怎么用

不一定,看這個例子,MapKey是一個不存在的屬性值,那么在Map里面就會存在一個Null,這是Hashmap決定的。

關(guān)于“MyBatis中@MapKey怎么用”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學(xué)到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI