SpringAOP联盟(1)—Advisor,Advice,Pointcut,Advised、ProxyConfig

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 在目标方法执行后执行,主要实现类:AspectJAfterAdviceAspectJAfterReturningAdviceAspectJAfterThrowingAdvice
AspectJAroundAdvice 环绕通知

在Proxy中最终执行的其实就是MethodInterceptor。因为这些Advice最终都是交给AdvisorAdapteradvice适配为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;
}

文章参考

https://blog.csdn.net/f641385712/article/details/89312652

你可能感兴趣的:(SpringAOP联盟(1)—Advisor,Advice,Pointcut,Advised、ProxyConfig)