96SEO 2026-02-19 22:31 15
动态代理功能生成一个Proxy代理类Proxy代理类实现了业务接口而通过调用Proxy代理类实现的业务接口实际上会触发代理类的invoke增强处理方法。

责任链功能可以动态地组合处理者增加或删除处理者而不需要修改客户端代码可以灵活地处理请求每个处理者可以选择处理请求或将请求传递给下一个处理者。
org.mybatis.spring.boot.autoconfigure.MybatisAutoConfigurationorg.springframework.context.annotation.Configuration
ConditionalOnSingleCandidate(DataSource.class)
EnableConfigurationProperties(MybatisProperties.class)
DataSourceAutoConfiguration.class,
MybatisLanguageDriverAutoConfiguration.class
MybatisAutoConfiguration(MybatisProperties
interceptorsProvider,ObjectProviderTypeHandler[]
languageDriversProvider,ResourceLoader
ObjectProviderDatabaseIdProvider
databaseIdProvider,ObjectProviderListConfigurationCustomizer
configurationCustomizersProvider)
interceptorsProvider.getIfAvailable();......}BeanConditionalOnMissingBeanpublic
SqlSessionFactoryBean();factory.setDataSource(dataSource);factory.setVfs(SpringBootVFS.class);if
(StringUtils.hasText(this.properties.getConfigLocation()))
{factory.setConfigLocation(this.resourceLoader.getResource(this.properties.getConfigLocation()));}applyConfiguration(factory);if
(this.properties.getConfigurationProperties()
{factory.setConfigurationProperties(this.properties.getConfigurationProperties());}//
(!ObjectUtils.isEmpty(this.interceptors))
{factory.setPlugins(this.interceptors);}......return
【1】interceptorsProvider.getIfAvailable();获取Interceptor接口的所有的bean并加载到内存interceptors里
【2】构造SqlSessionFactoryBean会用到内存的interceptors填充拦截器bean到SqlSessionFactoryBean里面。
SqlSessionFactoryBean#afterPropertiesSet初始化完成后执行
因为SqlSessionFactoryBean实现了InitializingBean接口必然有一个afterPropertiesSet()的实现方法
ApplicationListenerApplicationEvent
required);notNull(sqlSessionFactoryBuilder,
【3】这里会构造一个sqlSessionFactory并调用了buildSqlSessionFactory()
SqlSessionFactoryBean#buildSqlSessionFactory构造SqlSessionFactory
里面通过遍历SqlSessionFactoryBean的interceptors逐个把拦截器加载到SqlSessionFactory的interceptorChain里面。
{Stream.of(this.plugins).forEach(plugin
{targetConfiguration.addInterceptor(plugin);LOGGER.debug(()
this.sqlSessionFactoryBuilder.build(targetConfiguration);
【4】逐个把拦截器加载到targetConfiguration对象的interceptorChain里面也就是拦截器责任链了。
【5】最终通过sqlSessionFactoryBuilder建造者模式完成一个对象创建new
DefaultSqlSessionFactory(config)
DefaultSqlSessionFactory(Configuration
到此完成Mybatis的插件bean的类加载和插件责任链的初始化。
上述的实现逻辑拆分到了只包含部分逻辑的、功能单一的Handler处理类里开发人员可以按照业务需求将多个Handler对象组合成一条责任链实现请求的处理。
自然是每个sqlSession创建时在返回Executor对象前会对执行器进行一个pluginAll的插件处理。
我们发起一次请求最终会映射打开一个session会话http://localhost:8891/user_select?id2
DefaultSqlSessionFactory#openSessionFromDataSource
最终debug到org.apache.ibatis.session.defaults.DefaultSqlSessionFactory#openSessionFromDataSource
openSessionFromDataSource(ExecutorType
configuration.getEnvironment();final
getTransactionFactoryFromEnvironment(environment);//
transactionFactory.newTransaction(environment.getDataSource(),
DefaultSqlSession(configuration,
ExceptionFactory.wrapException(Error
{ErrorContext.instance().reset();}}
【1】获取环境变量【2】创建新事务【3】创建一个新的Executor执行器非常关键【4】返回新构造的DefaultSqlSession
CachingExecutor(executor);}executor
interceptorChain.pluginAll(executor);return
interceptorChain.pluginAll(executor)
interceptor.plugin(target);}return
是一个静态方法本质是对插件进行动态代理最终返回的是一个动态代理了Plugin类的自动生产对象。
org.apache.ibatis.plugin.Plugin#wrap
org.apache.ibatis.plugin.Plugin#wrappublic
getSignatureMap(interceptor);Class?
Proxy.newProxyInstance(type.getClassLoader(),interfaces,new
targetExecutor执行器Interceptor拦截器signatureMap拦截签名new
了一个Plugin类构造对Plugin类的动态代理会自动生成一个代理类A#Plugin最终执行的还是Plugin类的invoke方法于是wrap最终返回的是一个动态代理了Plugin类的自动生产对象。
代理器是Plugin被代理类是targetExecutor或Handler
org.mybatis.spring.SqlSessionTemplate.SqlSessionInterceptor#invoke
最终执行的逻辑是调用sqlSession去接args参数这个反射执行的方法是
SqlSession.selectOne(java.lang.String,java.lang.Object)
DefaultSqlSession#selectList(java.lang.String,
org.apache.ibatis.session.RowBounds)
configuration.getMappedStatement(statement);return
ExceptionFactory.wrapException(Error
{ErrorContext.instance().reset();}
executor.query是最终的Executor执行器query方法逻辑。
还记得上面一个步骤吗自动生成一个代理类A#Plugin它实现了对Executor的增强处理这块增强处理逻辑要回到Plugin#invoke方法
signatureMap.get(method.getDeclaringClass());if
ExceptionUtil.unwrapThrowable(e);}
这里会根据signatureMap的调用点来判断是否执行interceptor的拦截逻辑。
而这里的拦截逻辑就是我们实现好的Interceptor拦截器类的intercept方法啦。
new好的Invocation实际就是调用点。
通过InterceptorChain拦截器链对Executor进行增强
从图中可以知道Mybatis的拦截器链运用了动态代理和责任链模式
其实就是代理对象再次生成代理对象特殊的是代理对象的target属性另外配合Invocation类中的proceed方法形成责任链路的调用这样就可以在我们执行sql的前后做一些特殊的自定义的事情了。
作为专业的SEO优化服务提供商,我们致力于通过科学、系统的搜索引擎优化策略,帮助企业在百度、Google等搜索引擎中获得更高的排名和流量。我们的服务涵盖网站结构优化、内容优化、技术SEO和链接建设等多个维度。
| 服务项目 | 基础套餐 | 标准套餐 | 高级定制 |
|---|---|---|---|
| 关键词优化数量 | 10-20个核心词 | 30-50个核心词+长尾词 | 80-150个全方位覆盖 |
| 内容优化 | 基础页面优化 | 全站内容优化+每月5篇原创 | 个性化内容策略+每月15篇原创 |
| 技术SEO | 基本技术检查 | 全面技术优化+移动适配 | 深度技术重构+性能优化 |
| 外链建设 | 每月5-10条 | 每月20-30条高质量外链 | 每月50+条多渠道外链 |
| 数据报告 | 月度基础报告 | 双周详细报告+分析 | 每周深度报告+策略调整 |
| 效果保障 | 3-6个月见效 | 2-4个月见效 | 1-3个月快速见效 |
我们的SEO优化服务遵循科学严谨的流程,确保每一步都基于数据分析和行业最佳实践:
全面检测网站技术问题、内容质量、竞争对手情况,制定个性化优化方案。
基于用户搜索意图和商业目标,制定全面的关键词矩阵和布局策略。
解决网站技术问题,优化网站结构,提升页面速度和移动端体验。
创作高质量原创内容,优化现有页面,建立内容更新机制。
获取高质量外部链接,建立品牌在线影响力,提升网站权威度。
持续监控排名、流量和转化数据,根据效果调整优化策略。
基于我们服务的客户数据统计,平均优化效果如下:
我们坚信,真正的SEO优化不仅仅是追求排名,而是通过提供优质内容、优化用户体验、建立网站权威,最终实现可持续的业务增长。我们的目标是与客户建立长期合作关系,共同成长。
Demand feedback