通过前面的debug,我们知道了SqlSessionFactory和SqlSession两个对象是如何构建的,下面我看一下如何从sqlSession中获取mapper对象,也就是这一行:
UserMapper mapper = session.getMapper(UserMapper.class);
首先,定位起点:
1、UserMapper mapper = session.getMapper(UserMapper.class);
这里sqlSession的默认实现是DefaultSqlSession,所以,getMapper方法我们定位到这里类里面:
1、UserMapper mapper = session.getMapper(UserMapper.class);
2、org.apache.ibatis.session.defaults.DefaultSqlSession#getMapper
我们的mapper都是要在全局配置中配置的,所以这里可以看到从配置对象中获取mapper,
1、UserMapper mapper = session.getMapper(UserMapper.class);
2、org.apache.ibatis.session.defaults.DefaultSqlSession#getMapper
3、org.apache.ibatis.session.Configuration#getMapper
还记得我们的 MyBatis源码分析(02)Configuration和SqlSessionFactory 一节中,解析全局配置中的单个元素,比如mapper的解析过程,即使这一步:
org.apache.ibatis.builder.xml.XMLConfigBuilder#mapperElement
我们进入上面这个方法查看:
最后一行就是 configuration.addMapper(mapperInterface); ,这个是把mapper放到哪里了呢?我们进入查看一下:
这里可以清晰的看到,就是放到mapperRegistry里面了,这也是为什么我们上面能从mapperRegistry里面get到mapper。我们继续进入这个方法:
可以看到mapper最终是进入到了knownMappers中。类型作为key,代理工厂对象作为一个value放进去。
继续看第三步,获取mapper传入的是配置mapper的类型和sqlSession,下一步从mapperRegistry对象中获取mapper,无论是什么形式配置的mapper都会注册到这个里面:
1、UserMapper mapper = session.getMapper(UserMapper.class);
2、org.apache.ibatis.session.defaults.DefaultSqlSession#getMapper
3、org.apache.ibatis.session.Configuration#getMapper
4、org.apache.ibatis.binding.MapperRegistry#getMapper
我们平时写的mapper都是接口,但是我们调用却能获取结果,是因为mybatis使用了动态代理,给我们动态生成了实现对象,从上面的代码可以看出,这里根据传入的mapper类型构建了一个代理工厂对象,然后开始创建mapper,下面的mapperProxyFactory.newInstance(sqlSession);代码就是创建对象代码,来看下一步:
1、UserMapper mapper = session.getMapper(UserMapper.class);
2、org.apache.ibatis.session.defaults.DefaultSqlSession#getMapper
3、org.apache.ibatis.session.Configuration#getMapper
4、org.apache.ibatis.binding.MapperRegistry#getMapper
5、org.apache.ibatis.binding.MapperProxyFactory#newInstance(org.apache.ibatis.session.SqlSession)
6、org.apache.ibatis.binding.MapperProxyFactory#newInstance(org.apache.ibatis.binding.MapperProxy<T>)
这里就比较眼熟了,用过动态代理的都知道,这是jdk自带的动态代理的实现,根据我们调用时传入的接口类型,生成代理,然后创建代理对象,所以我们能获得UserMapper对象,操作里面的方法。