Spring的核心是IOC与AOP,IOC主要是依赖关系的管理,依赖查询,依赖注入,在之前bean的生命周期文章中已经对bean的生命周期做了相对多的分析,这里以前探讨下AOP的实现原理。
·AspectJAwareAdvisorAutoProxyCreator Spring启动时的beanPostProcessor,bean生命周期中经常可以看到beanPostProcessor在起作用,AOP也是基于IOC来实现的。
·DefaultAopProxyFactory.createAopProxy(AdvisedSupport config) 创建AOP代理类,这样才可以实现AOP
·AopNamespaceHandler aop标签解析的handler
·ConfigBeanDefinitionParser ,aop config标签的解析类。
1. 首先搭建一个能运行 Spring AOP的demo,参考早期的一篇文章Spring AOP入门
预览一下配置文件
2. 运行方式不变,在Bean生命周期-实例化ApplicationContext中的运行方式。
3. spring在启动的时候加载配置文件的方式不变,变化的只是其解析配置文件的方式,而实例化ApplicationContext中加载配置文件过程
refresh
- obtainFreshBeanFactory
- refreshBeanFactory
- AbstractRefreshableApplicationContext.loadBeanDefinitions(beanFactory)
- …
- XmlBeanDefinitionReader.loadBeanDefinitions(Resource resource)
- DefaultBeanDefinitionDocumentReader.parseBeanDefinitions
以上过程在bean生命周期中都有提到过,以上没有变化。
区别在于aop:config是自定义的标签,这次走的是parseCustomElement方法
4. 找到自定义元素的NamespaceHandler,NamespaceHandler的映射关系定义在META/spring.handlers文件中。Spring各个jar包中的spring.handlers都会生效。
NameSpaceHandler的映射关系存储在DefaultNamespaceHandlerResolver中,自定义的aop:config标签找到的Handler是AopNamespaceHandler
5. AopNamespaceHandler是在namespaceHandlerResolver进行resolve的时候,实例化并且调用init方法的,然后AopNamespaceHandler进行parse。
AopNamespaceHandler在parse的时候,找到标签对应的Parser再进行parse,因此[aop:config]标签又交给了ConfigBeanDefinitionParser来解析。
6. ConfigBeanDefinitionParser的解析过程,其刚开始配置configureAutoProxyCreator,会配置AspectJAwareAdvisorAutoProxyCreator这个类做为spring的bean。
ConfigBeanDefinitionParser.parse
- configureAutoProxyCreator
- AopNamespaceUtils.registerAspectJAutoProxyCreatorIfNecessary(parserContext, element)
- …
7. 在解析其它标签的时候,大致都是注册一个bean,pointcut,advisor,aspect内部解析就不相信说明了,这次我们只是概览。
8. 解析完aop的元素之后,其它的过程仍然和bean生命周期相同,不同之处则在于如果配置了AOP相关标签,在处理bean的时候,beanPostProcessor在bean生命周期内多处有影响,足以改变bean的结构。
AOP的实现方式基于IOC,通过beanPostProcessor来自定义bean的结构
9. 在bean实例化完成之后,会调用beanPostProcessor的postProcessAfterInitialization方法,这个生命阶段,我们在Bean生命周期汇总这篇文章里也提到过。
AspectJAwareAdvisorAutoProxyCreator的postProcessAfterInitialization方法再其父类AbstractAutoProxyCreator中实现,其会对已经实例化的bean进行wrap。
而对bean进行wrap的时候,就是使用ProxyFactory createAopProxy,最终会走到DefaultAopProxyFactory的createAopProxy方法。
可以看到:
·如果bean的类是接口或者类是JDK内部的代理类,其使用 JDK的动态代理类
·其它情况是CGLIB来实现
10. 最后生成的HelloWorld长这个样子。其已经是生成的代理类了,AOP功能已经生效。
11. 创建代理类之后,其余的过程与bean的生命周期基本一致
aop的功能也是借助spring对bean的管理来实现的,弄明白了bean的整个过程,spring的其它模块理解起来都会很轻松。