什么是AOP
Aspect-oriented Programming (AOP) complements
Object-oriented Programming (OOP) by providing another way of thinking about program structure.
The key unit of modularity in OOP is the class,
whereas in AOP the unit of modularity is the aspect.
Aspects enable the modularization of concerns (such
as transaction management) that cut across multiple
types and objects. (Such concerns are often termed “crosscutting” concerns in AOP literature.)
Spring官网上说AOP是对OOP的补充,在OOP中模块化的关键部分是类,而AOP中模块化的关键部分是切面。
Spring的AOP
基于接口实现AOP
Spring老的版本实现AOP是通过实现Advisor接口,并通过ProxyFactoryBean来生成一个代理对象。
AspectJ
AspectJ为实现AOP提供了注解的方式,并且其切点可以细粒度的控制,于是Spring就借助类AspectJ的这些注解,以及Aspect解析切面的能力。
Spring-AOP底层原理
Spring官网说,Spring的一个关键组件是AOP框架。虽然Spring IoC容器不依赖于AOP(这意味着如果您不想使用AOP就不需要使用),但AOP补充了Spring IoC,提供了一个功能非常强大的中间件解决方案。
官网说AOP补充了Spring IoC,而我本人理解AOP则是Spring IoC这个模型的一个典型的应用。
开发者在开启AOP时需要使用@EnableAspectJAutoProxy注解,该注解通过Import的方式为Spring容器注册了一个BeanDefinition—AspectJAwareAdvisorAutoProxyCreator,如下图所示,AspectJAwareAdvisorAutoProxyCreator最终实现了BeanPostProcessor接口,这就是本人理解AOP是Spring IoC模型的一个典型应用的原因。
解析切面
在Spring容器创建第一个bean时,IoC容器回调Bean的后置处理器执行postProcessBeforeInstantiation()方法,从容器中获取所有BeanDefinition,判断BD的类上是否加了@AspectJ注解,如果加了就进行解析,反射获取该类的所有方法,如果加了@Before、@After、@AfterReturning、@AfterThrowing注解就生成一个Advisor并缓存起来。(Advisor就是通知+切点)
解析切面是比较耗时的操作,因为需要把所有的BD拿出来,但是Spring为什么还要每创建一个Bean都调用一次呢,这是为了防止此时有新加入的BeanDefinition得不到解析,而Spring在这里做了缓存,所以并不会重复解析。
创建代理
创建Bean并完成初始化之后,IoC容器回调Bean后置处理器的postProcessAfterInitialization()方法。
调用wrapIfNecessary(bean, beanName, cacheKey),会先去筛选可以应用在该Bean上的Advisor并排好序。
经过一系列校验,如果需要代理,接下里创建该Bean的代理对象。
责任链调用
这里的ExposeInvocationInterceptor,再创建代理对象之前,找到所有匹配的Advisor放入一个list之后,IoC容器会向匹配的Advisor的list添加这个Advisor。