Spring AOP

Posted ā mén , Can i have a pen?

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Spring AOP相关的知识,希望对你有一定的参考价值。

此文主要是对于spring中AOP实现的一些思考总结 图。

 

一、先认识两个AOP功能类

  1、根据一个表达式判断一个Method是否匹配。

  2、根据拦截类类名和拦截器的方法名,获取指定拦截器方法Method。(拦截器即Advice)

其实,在Spring中,还对MethodLocatingFactory提出了一个更高层次的抽象FactoryBean(无非就是说,这个Bean是一个工厂Bean,它的功能是做其他事情,而不是直接供用户使用。比如我们现在的MethodLocatingFactory就是用于生成一个指定的Method)。如下图(AspectInstanceFactory暂忽略,后面生成Advice会用到):

补充:面试中可能会问到BeanFactory和FactoryBean的区别?

二、讲解一个策略ReflectiveMethodInvocation(重要!重要!重要!)

  一个可以实现按一定规则(before、afterReturning、afterThrowing等)链式调用增强方法的策略ReflectiveMethodInvocation。如下图:

 

三、代理工厂AopProxyFactory

  对以上的封装:
     AopConfig<-AopConfigSupport:包含有:advice拦截器集合、target目标对象。
        主要作用是提供一个方法:通过指定的方法获取其匹配的拦截器集合List<MethodInterceptor>。封装到DynamicAdvisedInterceptor中,最终传递给  ReflectiveMethodInvocation去链式处理。  
  最后封装到AopProxyFactory中,如下图所示:

四、拦截器Advice的合成

  AOP中的重中之重,说白了就是根据配置文件生成拦截器,然后通过CGLIB或者JDK生成出代理对象。

  (CGLIB和JDK动态代理,可戳此文https://www.cnblogs.com/zomicc/p/12255510.html

  Advice的建立依赖于PointCut,先引入PointCut,PointCut如下图:

  在Spring中,生成一个对象是通过BeanDefinition,Advice也不例外。
     先介绍下构造Advice对应的BeanDefinition和构造Advice都需要哪些信息:
      Advice(Method adviceMethod,AspectJExpressionPointcut pointcut,AspectInstanceFactory adviceObjectFactory)
         参数一:拦截器执行方法;
         参数二:切入点(顾名思义,用于指定拦截器在哪个TargetMethod上执行);
         参数三:切面初始化工厂(用于获取拦截器对象)。
      Advice对应的BeanDefinition构造参数都有:
         参数一:GenericBeanDefinition(Class类型是MethodLocatingFactory,作用是在生成Advice时获取Method对象。为什么不直接是Method,是因为在xml

          解析阶段不进行其他逻辑操作)
         参数二:RuntimeReference(对应切入点)
         参数三:GenericBeanDefinition(Class类型是AspectInstanceFactory)

    以AspectJBeforeAdvice为例,介绍合成BeanDefinition的原理,如下图所示:

至此,拦截器对应的BeanDefinition就已经定义好了,通过XmlBeanDefinitionReader解析即可将所有的合成BeanDefinition加入到BeanFactory。

然后就是使用Ioc中的构造器注入即可生产出真实的拦截器对象Advice。

需要注意的两点是:对于嵌套BeanDefinition需要进一步取值处理,对于FactoryBean类型的也需要获取到真实内容:(重要!重要!重要

 1 public Object resolveValueIfNecessary(Object value) {
 2         
 3         if (value instanceof RuntimeBeanReference) {
 4             RuntimeBeanReference ref = (RuntimeBeanReference) value;            
 5             String refName = ref.getBeanName();            
 6             Object bean = this.beanFactory.getBean(refName);                
 7             return bean;
 8             
 9         }else if (value instanceof TypedStringValue) {
10             return ((TypedStringValue) value).getValue();
11         } else if (value instanceof BeanDefinition) {//一、对于嵌套的BeanDefinition需要进一步取值
12             // Resolve plain BeanDefinition, without contained name: use dummy name.
13             BeanDefinition bd = (BeanDefinition) value;
14             
15             String innerBeanName = "(inner bean)" + bd.getBeanClassName() + "#" +
16                     Integer.toHexString(System.identityHashCode(bd));
17             
18             return resolveInnerBean(innerBeanName, bd);
19             
20         } 
21         else{
22             return value;
23         }    
24     }
25     private Object resolveInnerBean(String innerBeanName, BeanDefinition innerBd) {
26     
27         try {
28             
29             Object innerBean = this.beanFactory.createBean(innerBd);//二、获取内部BeanDefinition对应的Bean
30             
31             if (innerBean instanceof FactoryBean) {//三、如果是FactoryBean类型,则需要通过getObject()方法获取真实的内容
32                 try {
33                     return ((FactoryBean<?>)innerBean).getObject();
34                 } catch (Exception e) {                    
35                     throw new BeanCreationException(innerBeanName, "FactoryBean threw exception on object creation", e);
36                 }                
37             }
38             else {
39                 return innerBean;
40             }
41         }
42         catch (BeansException ex) {
43             throw new BeanCreationException(
44                     innerBeanName,
45                     "Cannot create inner bean \'" + innerBeanName + "\' " +
46                     (innerBd != null && innerBd.getBeanClassName() != null ? "of type [" + innerBd.getBeanClassName() + "] " : "")
47                     , ex);
48         }
49     }

五、最终通过Bean生命周期的“钩子”函数AspectJAutoProxyCreator生成真实的代理对象

  本质上就是通过BeanPostProcessor的afterInitialization()方法对AopProxyFactory的进一步封装。AspectJAutoProxyCreator代码如下:

public class AspectJAutoProxyCreator implements BeanPostProcessor {
    ConfigurableBeanFactory beanFactory;
    public Object beforeInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }

    public Object afterInitialization(Object bean, String beanName) throws BeansException {
        
        //如果这个Bean本身就是Advice及其子类,那就不要再生成动态代理了。
        if(isInfrastructureClass(bean.getClass())){
            return bean;
        }
        
        List<Advice> advices = getCandidateAdvices(bean);//获取可用拦截器
        if(advices.isEmpty()){
            return bean;
        }
        
        return createProxy(advices,bean);
    }
    
    private List<Advice> getCandidateAdvices(Object bean){
        
        List<Object> advices = this.beanFactory.getBeansByType(Advice.class);
        
        List<Advice> result = new ArrayList<Advice>();
        for(Object o : advices){            
            Pointcut pc = ((Advice) o).getPointcut();
            if(canApply(pc,bean.getClass())){
                result.add((Advice) o);
            }
            
        }
        return result;
    }
    
    protected Object createProxy( List<Advice> advices ,Object bean) {
        
        
        AopConfigSupport config = new AopConfigSupport();
        for(Advice advice : advices){
            config.addAdvice(advice);//构造AopConfig(添加拦截器)
        }
        
        Set<Class> targetInterfaces = ClassUtils.getAllInterfacesForClassAsSet(bean.getClass());
        for (Class<?> targetInterface : targetInterfaces) {
            config.addInterface(targetInterface);
        }
        
        config.setTargetObject(bean);//构造AopConfig(添加目标对象)     
        
        AopProxyFactory proxyFactory = null;
        if(config.getProxiedInterfaces().length == 0){
            proxyFactory =  new CglibProxyFactory(config);//cglib代理
        } else{
            
            proxyFactory = new JdkAopProxyFactory(config);
        }  
        
        return proxyFactory.getProxy();//获取代理对象
        
        
    }
    
    protected boolean isInfrastructureClass(Class<?> beanClass) {
        boolean retVal = Advice.class.isAssignableFrom(beanClass);
        
        return retVal;
    }
    
    public void setBeanFactory(ConfigurableBeanFactory beanFactory) {
        this.beanFactory = beanFactory;
        
    }
    
    public static boolean canApply(Pointcut pc, Class<?> targetClass) {
        
        MethodMatcher methodMatcher = pc.getMethodMatcher();    

        Set<Class> classes = new LinkedHashSet<Class>(ClassUtils.getAllInterfacesForClassAsSet(targetClass));
        classes.add(targetClass);
        for (Class<?> clazz : classes) {
            Method[] methods = clazz.getDeclaredMethods();            
            for (Method method : methods) {
                if (methodMatcher.matches(method/*, targetClass*/)) {//有一个方法与切入点匹配,则该bean就是需要动态代理的对象
                    return true;
                }
            }
        }
        return false;
    }

}

 

 

  最后,Bean生命周期图附一张:

 

AOP就这么多了,你品,你细品!

 

以上是关于Spring AOP的主要内容,如果未能解决你的问题,请参考以下文章

JAVA之AOP

Spring AOP

Spring源码高级笔记之——Spring AOP应用

2018.12.24 Spring中的aop演示

Spring框架 AOP

Spring的AOP面向切面编程