Spring系列之AOP分析之获取Advisor的过程续(三)

我们在这一章中继续上一章的分析。我们在上一章中说到了获取到了切面类中所有不带@Pointcut注解的方法,我们看看Spring对我们获取到的这些方法进行了什么操作:

for (Method method : getAdvisorMethods(aspectClass)) {
    //循环切面中所有不带@Pointcut注解的方法。 
    //method 切面中不带@Pointcut注解的方法
    //lazySingletonAspectInstanceFactory中含有切面实例 是个单例
    //advisors中的下标 切面的一个顺序
    //aspectName 切面类的名字
    //下面我们来看一下getAdvisor这个方法的内容
    Advisor advisor = getAdvisor(method, lazySingletonAspectInstanceFactory, advisors.size(), aspectName);
    if (advisor != null) {
        advisors.add(advisor);
    }
}

getAdvisor

    public Advisor getAdvisor(Method candidateAdviceMethod, MetadataAwareAspectInstanceFactory aspectInstanceFactory,
            int declarationOrderInAspect, String aspectName) {
        //验证切面类 见我们上一章的分析
        validate(aspectInstanceFactory.getAspectMetadata().getAspectClass());
        //这里根据传入的方法和切面类获取 切点表达式
        AspectJExpressionPointcut expressionPointcut = getPointcut(
                candidateAdviceMethod, aspectInstanceFactory.getAspectMetadata().getAspectClass());
        //没有获取到切点表达式 直接返回null
        if (expressionPointcut == null) {
            return null;
        }
        //返回一个Advisor的实例 这个实例中包含了 一下内容
        //切点表达式 AspectJExpressionPointcut 
        //切点方法 
        //ReflectiveAspectJAdvisorFactory实例
        //切面类实例
        //切面类名字
        return new InstantiationModelAwarePointcutAdvisorImpl(expressionPointcut, candidateAdviceMethod,
                this, aspectInstanceFactory, declarationOrderInAspect, aspectName);
    }

我们先来看一下AspectJExpressionPointcut这个类的UML类图:
Spring系列之AOP分析之获取Advisor的过程续(三)
在AspectJExpressionPointcut这个类中主要实现了以下这四大接口:ClassFilter、BeanFactoryAware、MethodMatcher、Pointcut。Pointcut是SpringAOP中定义的接口,用来表示切面的抽象。BeanFactoryAware也是SpringIOC中一个常见的接口,用来设置BeanFactory实例。ClassFilter和MethodMatcher是SpringAOP中定义的进行Advisor匹配的接口。ClassFilter用来此Advisor是否使用于目标类。MethodMatcher用来匹配此Advisor是否可以作用于目标类中的目标方法。那么AspectJExpressionPointcut这个类就拥有了一下功能:从BeanFactory中获取Bean、拥有切点表达式、可以用来判断此切点表达式方法是否适用于目标类、此切点表达式方法是否适用于目标类中的方法。OK,我们继续看生成AspectJExpressionPointcut的过程。

    private AspectJExpressionPointcut getPointcut(Method candidateAdviceMethod, Class<?> candidateAspectClass) {
        //方法上是否存在切面注解(通知类型注解) 即方法上是否有
        //@Before, @Around, @After, @AfterReturning, @AfterThrowing, @Pointcut注解
        //这里也提供了一种获取类上是否有我们想要的注解的一种方式
        //返回一个AspectJAnnotation对象
        //这里用了AnnotationUtils用来获取注解的相关信息
        AspectJAnnotation<?> aspectJAnnotation =
                AbstractAspectJAdvisorFactory.findAspectJAnnotationOnMethod(candidateAdviceMethod);
        if (aspectJAnnotation == null) {
            return null;
        }
        //创建AspectJExpressionPointcut对象
        AspectJExpressionPointcut ajexp =
                new AspectJExpressionPointcut(candidateAspectClass, new String[0], new Class<?>[0]);
        //设置切面表达式 在AspectJAnnotation中是可以获取到通知类型的,但是这里没有设置通知类型
        ajexp.setExpression(aspectJAnnotation.getPointcutExpression());
        ajexp.setBeanFactory(this.beanFactory);
        return ajexp;
    }

在这里我们要看一下AspectJAnnotation这个类。这个类是AbstractAspectJAdvisorFactory中的内部类。从这个类中可以获取切点表达式和通知类型。

    protected static class AspectJAnnotation<A extends Annotation> {
        //切点表达式所在的属性  poincut会覆盖value的值 
        //其实这里指的是 通知类型注解中的属性
        private static final String[] EXPRESSION_PROPERTIES = new String[] {"value", "pointcut"};

        private static Map<Class<?>, AspectJAnnotationType> annotationTypes =
                new HashMap<Class<?>, AspectJAnnotationType>();

        static {
            //初始化通知类型
            annotationTypes.put(Pointcut.class,AspectJAnnotationType.AtPointcut);
            annotationTypes.put(After.class,AspectJAnnotationType.AtAfter);
            annotationTypes.put(AfterReturning.class,AspectJAnnotationType.AtAfterReturning);
            annotationTypes.put(AfterThrowing.class,AspectJAnnotationType.AtAfterThrowing);
            annotationTypes.put(Around.class,AspectJAnnotationType.AtAround);
            annotationTypes.put(Before.class,AspectJAnnotationType.AtBefore);
        }
        .......
        public AspectJAnnotation(A annotation) {
            //注解信息
            this.annotation = annotation;
            //根据注解类型获取通知类型
            this.annotationType = determineAnnotationType(annotation);
            // We know these methods exist with the same name on each object,
            // but need to invoke them reflectively as there isn't a common interface.
            try {
                //从通知类型注解上面获取切点表达式
                this.pointcutExpression = resolveExpression(annotation);
                //获取参数的名字
                this.argumentNames = (String) annotation.getClass().getMethod("argNames").invoke(annotation);
            }
            catch (Exception ex) {
                throw new IllegalArgumentException(annotation + " cannot be an AspectJ annotation", ex);
            }
        }
        //从这个获取切点表达式的代码中我们可以看到 pointcut的属性会覆盖value的属性值
        private String resolveExpression(A annotation) throws Exception {
            String expression = null;
            //循环上面的切点属性
            for (String methodName : EXPRESSION_PROPERTIES) {
                Method method;
                try {
                    method = annotation.getClass().getDeclaredMethod(methodName);
                }
                catch (NoSuchMethodException ex) {
                    method = null;
                }
                if (method != null) {
                    //获取切点表达式!!!
                    String candidate = (String) method.invoke(annotation);
                    if (StringUtils.hasText(candidate)) {
                        expression = candidate;
                    }
                }
            }
            return expression;
        }

我们把上面获取Advisor的过程总结一下:循环切面类中的所有不带@Pointcut注解的方法,接着判断切面类的方法上是否有:@Before, @Around, @After, @AfterReturning, @AfterThrowing, @Pointcut注解。如果没有的话,循环下一个方法。如果有这些注解的话,则从这些注解中获取切点表达式存放到AspectJExpressionPointcut对象中,最后将获取到的切点表达式类封装到InstantiationModelAwarePointcutAdvisorImpl这个类中。从上面的分析我们知道InstantiationModelAwarePointcutAdvisorImpl类至少拥有:切点表达式类、切面对象、带有切点表达式的方法、ReflectiveAspectJAdvisorFactory实例。那么我们最后来分析一下InstantiationModelAwarePointcutAdvisorImpl这个类。

    public InstantiationModelAwarePointcutAdvisorImpl(AspectJExpressionPointcut declaredPointcut,
            Method aspectJAdviceMethod, AspectJAdvisorFactory aspectJAdvisorFactory,
            MetadataAwareAspectInstanceFactory aspectInstanceFactory, int declarationOrder, String aspectName) {
        //切点表达式类 AspectJExpressionPointcut 
        this.declaredPointcut = declaredPointcut;
        //切面类 
        this.declaringClass = aspectJAdviceMethod.getDeclaringClass();
        //切点表达式方法所在的方法名 这里指的是@Before、@After这些通知类型所在的方法名
        this.methodName = aspectJAdviceMethod.getName();
        //通知参数类型
        this.parameterTypes = aspectJAdviceMethod.getParameterTypes();
        //切面通知方法
        this.aspectJAdviceMethod = aspectJAdviceMethod;
        //ReflectiveAspectJAdvisorFactory实例
        this.aspectJAdvisorFactory = aspectJAdvisorFactory;
        //切面对象实例
        this.aspectInstanceFactory = aspectInstanceFactory;
        //切面顺序
        this.declarationOrder = declarationOrder;
        this.aspectName = aspectName;
        //根据切面元数据判断是否要延迟实例化 一般为否
        if (aspectInstanceFactory.getAspectMetadata().isLazilyInstantiated()) {
            // Static part of the pointcut is a lazy type.
            Pointcut preInstantiationPointcut = Pointcuts.union(
                    aspectInstanceFactory.getAspectMetadata().getPerClausePointcut(), this.declaredPointcut);
            this.pointcut = new PerTargetInstantiationModelPointcut(
                    this.declaredPointcut, preInstantiationPointcut, aspectInstanceFactory);
            this.lazy = true;
        }
        else {
            // A singleton aspect.
            //切点表达式类
            this.pointcut = this.declaredPointcut;
            this.lazy = false;
            //这里获取Advice实例 这里又拥有了Advice的实例!!!!
            //不得不说InstantiationModelAwarePointcutAdvisorImpl这个类真的是太强大了
            this.instantiatedAdvice = instantiateAdvice(this.declaredPointcut);
        }
    }

OK下面我们的分析就到了如果从InstantiationModelAwarePointcutAdvisorImpl中获取Advice的内容。

上一篇:数据库的基本概念


下一篇:uniapp 如何在手机关屏后仍能正常运行、接收信息