原文:SpringAOP联盟(1)—Advisor,Advice,Pointcut,Advised、ProxyConfig - 简书 (jianshu.com)
代理对象生成
@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(); }
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 通知
// 有点标记接口的意思 public interface Advice { }
Advice通知 | 作用 |
MethodBeforeAdvice | 在目标方法执行之前执行。主要实现:AspectJMethodBeoreAdvice
|
AfterReturningAdvice | 在目标方法执行后执行,主要实现类:AspectJAfterAdvice 、AspectJAfterReturningAdvice 、AspectJAfterThrowingAdvice
|
AspectJAroundAdvice | 环绕通知 |
在Proxy中最终执行的其实就是MethodInterceptor
。因为这些Advice最终都是交给AdvisorAdapter
将advice
适配为MethodInterceptor
。
@FunctionalInterface public interface MethodInterceptor extends Interceptor { // 被代理后的逻辑 @Nullable Object invoke(@Nonnull MethodInvocation invocation) throws Throwable; }
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. Advicor
类名 | 作用 |
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设置为true也会被忽略。 private boolean optimize = false; //标记是否需要阻止通过该配置创建的代理对象转换为Advised类型,默认值为false,表示代理对象可以被转换为Advised类型 boolean opaque = false; //标记代理对象是否可以被AopContext以ThreadLocal的形式暴露出去。 boolean exposeProxy = false; //false:允许对代理对象进行修改(在Advisor链表中新增一个Advisor);true:不允许对代理对象进行修改。 private boolean frozen = false; }