SpringAop类比较错综复杂,但向上追述源头,那么可以理解为下图的关系。
关系图.png
源码介绍
案例:代理对象的生成过程
@Test
public void testProxyFactory() {
Person person = new Person();
//被建议的类,即面向目标类生成代理类
ProxyFactory proxyFactory = new ProxyFactory(person);
NameMatchMethodPointcut nameMatchMethodPointcut = new NameMatchMethodPointcut();
nameMatchMethodPointcut.addMethodName("run1");
//通知+切点=advisor
DefaultPointcutAdvisor advisor = new DefaultPointcutAdvisor();
advisor.setPointcut(nameMatchMethodPointcut);
advisor.setAdvice(new MethodBeforeAdvice() {
@Override
public void before(Method method, Object[] args, Object target) throws Throwable {
System.out.println("before Advice...");
}
});
//advisor放入到adviced
proxyFactory.addAdvisor(advisor);
//最后经过代理生成代理对象
Person proxy = (Person) proxyFactory.getProxy();
proxy.run1();
}
SpringAOP概述
AOP面向切面编程,在程序中具有公共特性的某些类/某些方法进行拦截,在方法执行前/执行后等增加一些方法。
1. Pointcut切点
类名 | 作用 |
---|---|
NameMatchMethodPointcut | 通过方法名进行精确匹配的。 |
ControlFlowPointcut | 根据在当前线程的堆栈信息中的方法名来决定是否切入某个方法(效率较低) |
ComposablePointcut | 组合模式的 Pointcut, 主要分成两种: 1.组合中所有都匹配算成功 2. 组合中都不匹配才算成功 |
JdkRegexpMethodPointcut | 通过 正则表达式来匹配方法 |
AspectJExpressionPointcut | 通过 AspectJ 包中的组件进行方法的匹配(切点表达式) |
TransactionAttributeSourcePointcut | 通过 TransactionAttributeSource 在 类的方法上提取事务注解的属性 @Transactional 来判断是否匹配, 提取到则说明匹配, 提取不到则说明匹配不成功 |
AnnotationJCacheOperationSource | 支持JSR107的cache相关注解的支持 |
// 由 ClassFilter 与 MethodMatcher 组成的 pointcut
public interface Pointcut {
// 类过滤器, 可以知道哪些类需要拦截
ClassFilter getClassFilter();
// 方法匹配器, 可以知道哪些方法需要拦截
MethodMatcher getMethodMatcher();
// 匹配所有对象的 Pointcut
Pointcut TRUE = TruePointcut.INSTANCE;
}
2. Advice通知
Advice通知 | 作用 |
---|---|
MethodBeforeAdvice | 在目标方法执行之前执行。主要实现:AspectJMethodBeoreAdvice
|
AfterReturningAdvice | 在目标方法执行后执行,主要实现类:AspectJAfterAdvice 、AspectJAfterReturningAdvice 、AspectJAfterThrowingAdvice
|
AspectJAroundAdvice | 环绕通知 |
在Proxy中最终执行的其实就是MethodInterceptor
。因为这些Advice最终都是交给AdvisorAdapter
将advice
适配为MethodInterceptor
。
MethodInterceptor | 作用 |
---|---|
CustomizableTraceInterceptor | 对方法调用前后拦截一下 |
SimpleTraceInterceptor | 正常效果同上,异常也是同样的输出,没CustomizableTraceInterceptor强大 |
DebugInterceptor | SimpleTraceInterceptor的子类。有个计数器,记录被拦截的次数,且可以这样获取出来advice.getCount() |
PerformanceMonitorInterceptor | 记录每个方法运行的时长 |
AfterReturningAdviceInterceptor | 这个类其实就是将 AfterReturningAdvice 包裹成 MethodInterceptor 的适配类, 而做对应适配工作的就是 AfterReturningAdviceAdapter |
MethodBeforeAdviceInterceptor | 这个类其实就是将 MethodBeforeAdvice 包裹成 MethodInterceptor 的适配类, 而做对应适配工作的就是 MethodBeforeAdviceAdapter |
ThrowsAdviceInterceptor | 这个类其实就是将 ThrowsAdvice 包裹成 MethodInterceptor 的适配类, 而做对应适配工作的就是 ThrowsAdviceAdapter |
TransactionInterceptor | 这个类就是大名鼎鼎的注解式事务的工具类, 这个类通过获取注解在方法上的 @Transactional 注解的信息来决定是否开启事务的 MethodInterceptor |
3. Advisor
类名 | 作用 |
---|---|
PointcutAdvisor | Spring 中常用的 Advisor, 包含一个 Pointcut 与一个 advice |
AspectJPointcutAdvisor | Spring 解析 aop 命名空间时生成的 Advisor, 对于这个类的解析是在 ConfigBeanDefinitionParser |
InstantiationModelAwarePointcutAdvisorImpl | Spring解析被 @AspectJ注解注释的类时生成的 Advisor, 而这个 Advisor中的 Pointcut与Advice都是由 ReflectiveAspectJAdvisorFactory 来解析生成的 |
TransactionAttributeSourceAdvisor | 一个基于 MethodInterceptor(其实是 TransactionInterceptor)与 TransactionAttributeSourcePointcut 的Advisor, 而这个类最常与 TransactionProxyFactoryBean使用 |
DefaultPointcutAdvisor | 最常用的 Advisor, 在使用编程式aop时, 很多时候会将 Advice / MethodInterceptor 转换成 DefaultPointcutAdvisor |
NameMatchMethodPointcutAdvisor | 使用 NameMatchPointcutAdvisor时创建的 Advisor, 主要是通过 方法名来匹配是否执行 Advice |
RegexpMethodPointcutAdvisor | 基于正则表达式来匹配 Pointcut 的 Advisor, 其中的 Pointcut 默认是 JdkRegexpMethodPointcut |
4. Adviced
类名 | 作用 |
---|---|
ProxyFactory | 这个类通过构造函数中的 proxyInterface/interceptor/targetSource 来创建代理对象(这个类是编程式 AOP 中最常用的对象) |
ProxyFactoryBean | 这个类是基于 FactoryBean 的 Proxy创建形式, 其通过代理的 Interface, targetSource 与指定的 interceptorNames 来创建对应的AopProxy, 最后生成对应的代理对象 |
AspectJProxyFactory | 将一个被 @Aspect 注解标示的类丢入其中, 变创建了对应的代理对象 |
5. ProxyConfig
代理对象的配置属性。
public class ProxyConfig implements Serializable {
//true:表示使用Cglib代理。false:表示使用JDK代理
private boolean proxyTargetClass = false;
//true:那么在生成代理对象之后,如果对代理配置进行了修改,已经创建的代理对象也不会获取修改之后的代理配置。
//如果exposeProxy设置为true,那么optimize设置weitrue也会被忽略。
private boolean optimize = false;
//标记是否需要阻止通过该配置创建的代理对象转换为Advised类型,默认值为false,表示代理对象可以被转换为Advised类型
boolean opaque = false;
//标记代理对象是否可以被AopContext以ThreadLocal的形式暴露出去。
boolean exposeProxy = false;
//false:允许对代理对象进行修改(在Advisor链表中新增一个Advisor);true:不允许对代理对象进行修改。
private boolean frozen = false;
}