通过mybatis的加载和sql语句的执行过程,咱们不难发现mybatis代码中大量使用了动态代理,比如:
1:Mapper接口代理类,mybatis为所有的mapper接口类都生成了代理接口类
2:SqlSessionTemplate的SqlSession属性
3:Executor的拦截器会生成代理类
4:StatementHandler拦截器会生产代理类
5:ParameterHandler拦截器会生产代理类
4:ResultSetHandler拦截器会生产代理类
接下来我们来看一下四大核心组件之Executor的动态代理生成拦截器的过程
Executor是SqlSession的重要组成部分,我们通过SqlSession对象的创建作为入口来分析Executor的创建过程,
通过上面的代码可以看到,最终创建Executor是在Configuration里,创建前面根据Executor的type创建出不同的Executor,如果开启了二级缓存的情况下会返回CachingExecutor,最后
executor = (Executor) interceptorChain.pluginAll(executor)这行代码通过拦截器链条为Executor创建的代理对象,我们看看pluginAll方法的具体内容
循环插件集合来生产代理对象
插件需要实现Interceptor的plugin方法来生产动态代理对象,如上图使用过了Plugin的静态方法warp来生成动态代理对象
如果操作之后生成了一个Executor的动态对象,但是他是做了多次的动态代理,如下图
调用Executor会从最外层(跟interceptorChain的顺序正好相反)依次调用invok方法,最后会调用真正的Executor的方法
最后mybatis的四大组件的插件都是基于此方式集成的。
还有一个问题interceptorChain里的插件对象是什么时候放进去的呢? 读者可以自己跟踪代码来区分
答案:实现插件方自己实现