⽆处不在的代理模式(二)--mybatis中mapper代理sqlsession

 

 我们写一个UserMapper接口,Mybatis会为该接口创建一个MapperProxy对象。

      @Test
    public void selectTest() {
        // 会话是一个级别的 ,不能跨线程
        // 缓存// 对像
        /**  
         * 基于Mapper接口代理调用
         */
        UserMapper mapper = sqlSession.getMapper(UserMapper.class);
        User user = mapper.selectByid(10);//最终通过mapperProxy也会调用到sqlSession.selectone();
        // 基于会话调用
        /**
         * 基于会话直接调用
         */
        String statementID = "com.code.read.pattern.mybatis.UserMapper.selectByid";
        user = sqlSession.selectOne(statementID,10);
    }

1、MapperProxy类的创建

    1.1 MapperRegistry类

     MapperRegistry是mapper的注册类,提供了注册一个包下面的所有Mapper接口。内部维护了一个map以mapper接口类为key,value是接口类对应的代理工厂。

 private final Map<Class<?>, MapperProxyFactory<?>> knownMappers = new HashMap<>();

          

   MapperRegistry其中重要的方法void addMapper(Class<T> type)方法简单的将type放进knownMappers里面。

         

public <T> void addMapper(Class<T> type) {
    if (type.isInterface()) {
      if (hasMapper(type)) {
        throw new BindingException("Type " + type + " is already known to the MapperRegistry.");
      }
      boolean loadCompleted = false;
      try {
        knownMappers.put(type, new MapperProxyFactory<>(type));//插入mapper和对应的代理工厂
        // It's important that the type is added before the parser is run
        // otherwise the binding may automatically be attempted by the
        // mapper parser. If the type is already known, it won't try.
        MapperAnnotationBuilder parser = new MapperAnnotationBuilder(config, type);
        parser.parse();//解析mapper中的方法以及statement。
        loadCompleted = true;
      } finally {
        if (!loadCompleted) {
          knownMappers.remove(type);
        }

 

待续--------

 

posted @ 2020-09-13 15:17  小南的歌  阅读(213)  评论(0编辑  收藏  举报