Spring AOP 实现

来源:互联网 发布:dreamweaver破解版mac 编辑:程序博客网 时间:2024/06/14 19:39
    Aop是Aspect-Oriented Programming(面向方面编程或面向切面编程)的简称。在Spring平台功能中,AOP是一个核心模块,Spring将AOP框架与IoC容器紧密集成,从而为使用AOP提供最大便利。
     AOP可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。当我们需要为分散的对象引入公共行为的时候,OOP则显得无能为力。也就是说,OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系。例如日志功能。日志代码往往水平地散布在所有对象层次中,而与它所散布到的对象的核心功能毫无关系。对于其他类型的代码,如安全性、异常处理和透明的持续性也是如此。这种散布在各处的无关的代码被称为横切(cross-cutting)代码,在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。
      AOP技术则恰恰相反,它利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为“Aspect”,即方面。所谓“方面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。
   Spring AOP实现中,使用Java语言来实现增强对象与切面增强应用,并为这两者的结合提供了配置环境。
1、常用Spring AOP 的两种方式实现
     通常使用Spring AOP主要有两种方式,一种是基于xml配置的纯POJO切面方式,另一种是通过@AspectJ注解驱动的切面方式。如果是纯POJO切面的配置AOP,我们需要在Spring容器xml资源定义文件中声明切面bean,并将切面、切点和通知配置信息通<aop:config>标签加以说明。如果是@AspectJ注解驱动的切面,我们知道为了支持注解,我们需要在配置文件中加入<aop:aspectj-autoproxy />标签。不论何种形式,Spring容器在处理切面时的思路是一致的,在Spring容器的初始化过程中,当载入、解析配置文件中的这些标签,都需要动态的通过自定义标签机制解析这些标签,从而生成相应的BeanDefine。
 1)Spring Aop 自定义标签处理
    在类AopNamespaceHandler中,源码如下:
public class AopNamespaceHandler extends NamespaceHandlerSupport {public AopNamespaceHandler() {    }public void init() {this.registerBeanDefinitionParser("config", new ConfigBeanDefinitionParser());        this.registerBeanDefinitionParser("aspectj-autoproxy", new AspectJAutoProxyBeanDefinitionParser());        this.registerBeanDefinitionDecorator("scoped-proxy", new ScopedProxyBeanDefinitionDecorator());        this.registerBeanDefinitionParser("spring-configured", new SpringConfiguredBeanDefinitionParser());}}
   在init()函数中,注册了ConfigBeanDefinitionParser()用于解析处理纯POJO切面方式<aop:config>标签,然后注册了AspectJAutoProxyBeanDefinitionParser()用于解析处理<aop:aspectj-autoproxy />标签。
2)注册Aop自动代理创建器
   不管是  ConfigBeanDefinitionParser还是AspectJAutoProxyBeanDefinitionParser,他们都是实现BeanDefinitionParse接口,主要的方法是BeanDefinition parse(Element elementParserContext parserContext)。

ConfigBeanDefinitionParser的parse()方法:
public BeanDefinition parse(Element element, ParserContext parserContext) {    CompositeComponentDefinition compositeDef = new CompositeComponentDefinition(element.getTagName(), parserContext.extractSource(element));    parserContext.pushContainingComponent(compositeDef); this.configureAutoProxyCreator(parserContext, element);    List childElts = DomUtils.getChildElements(element);    Iterator var5 = childElts.iterator();    while(var5.hasNext()) {        Element elt = (Element)var5.next();        String localName = parserContext.getDelegate().getLocalName(elt);        if("pointcut".equals(localName)) {            this.parsePointcut(elt, parserContext);        } else if("advisor".equals(localName)) {            this.parseAdvisor(elt, parserContext);        } else if("aspect".equals(localName)) {            this.parseAspect(elt, parserContext);        }    }    parserContext.popAndRegisterContainingComponent();    return null;}
 在该函数中首先注册AutoProxyCreator,然后根据不同切面元素解析成为响应的BeanDefinition。this.configureAutoProxyCreator(parserContextelement);最终会执行
public static BeanDefinition registerAspectJAutoProxyCreatorIfNecessary(BeanDefinitionRegistry registry, Object source) {return registerOrEscalateApcAsRequired(AspectJAwareAdvisorAutoProxyCreator.class, registry, source);}
注册AspectJAwareAdvisorAutoProxyCreator。

AspectJAutoProxyBeanDefinitionParser的parse()方法:
public BeanDefinition parse(Element element, ParserContext parserContext) { AopNamespaceUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(parserContext, element);    this.extendBeanDefinition(element, parserContext);    return null;}public static BeanDefinition registerAspectJAnnotationAutoProxyCreatorIfNecessary(BeanDefinitionRegistry registry, Object source) {return registerOrEscalateApcAsRequired(AnnotationAwareAspectJAutoProxyCreator.class, registry, source);}
可以看到,同样的注册AspectJAnnotationAutoProxyCreator,不同于之前的是,AspectJAnnotationAutoProxyCreator继承自AspectJAwareAdvisorAutoProxyCreator,添加了对于注解的处理。

3)AutoProxyCreator类继承体系

     通过类的继承体系,可以看到AutoProxyCreator类实现了BeanPostProcessor接口,其实际是一
InstantiationAwareBeanPostProcessor。在bean的生命周期中,可以知道,当容器注册了BeanPostProcessor类型的bean后,从容器中获取任何的bean,都将经过BeanPostProcessor相关方法的处理。那么,当一个bean,被切入切面,进行功能增强时,也是通过InstantiationAwareBeanPostProcessor相关方法对这个bean进行处理,具体就是通过代理方式,将切面植入方法调用流程。而这个处理方法就是 Object postProcessAfterInitialization(Object beanString beanName)。

public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {if(bean != null) {        Object cacheKey = this.getCacheKey(bean.getClass(), beanName);        if(!this.earlyProxyReferences.contains(cacheKey)) {return this.wrapIfNecessary(bean, beanName, cacheKey);}    }return bean;}
在this.wrapIfNecessary(beanbeanNamecacheKey)中封装bean生成代理对象返回。

4)生成代理相关类继承体系


   在生成代理对象时,通过ProxyFactory类Object getProxy(TargetSource targetSource)方法生成,targetSource是被代理的对象bean封装。
public static Object getProxy(TargetSource targetSource) {if(targetSource.getTargetClass() == null) {throw new IllegalArgumentException("Cannot create class proxy for TargetSource with null target class");} else {        ProxyFactory proxyFactory = new ProxyFactory();proxyFactory.setTargetSource(targetSource);proxyFactory.setProxyTargetClass(true);        return proxyFactory.getProxy();}}
最终调用DefaultAopProxyFactory类的AopProxy createAopProxy(AdvisedSupport config)方法:
public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {if(!config.isOptimize() && !config.isProxyTargetClass() && !this.hasNoUserSuppliedProxyInterfaces(config)) {return new JdkDynamicAopProxy(config);} else {        Class targetClass = config.getTargetClass();        if(targetClass == null) {throw new AopConfigException("TargetSource cannot determine target class: Either an interface or a target is required for proxy creation.");} else {return (AopProxy)(targetClass.isInterface()?new JdkDynamicAopProxy(config):new ObjenesisCglibAopProxy(config));}    }}
Spring提供了两种方式来生成代理对象: JDKProxy和Cglib,具体使用哪种方式生成由AopProxyFactory根据AdvisedSupport对象的配置来决定。默认的策略是如果目标类是接口,则使用JDK动态代理技术,否则使用Cglib来生成代理。
  而JdkDynamicAopProxy是标准的java动态代理使用方式,可以看到其实现了InvocationHandler接口。在JdkDynamicAopProxy中,通过Object getProxy(ClassLoader classLoader)函数,获取代理:
public Object getProxy(ClassLoader classLoader) {if(logger.isDebugEnabled()) {        logger.debug("Creating JDK dynamic proxy: target source is " + this.advised.getTargetSource());}    Class[] proxiedInterfaces = AopProxyUtils.completeProxiedInterfaces(this.advised);    this.findDefinedEqualsAndHashCodeMethods(proxiedInterfaces);    return Proxy.newProxyInstance(classLoader, proxiedInterfaces, this);}
我们知道InvocationHandler是JDK动态代理的核心,生成的代理对象的方法调用都会委托到InvocationHandler.invoke()方法。
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {    Object oldProxy = null;    boolean setProxyContext = false;TargetSource targetSource = this.advised.targetSource;Class targetClass = null;Object target = null;Object retVal;    try {      //eqauls()方法,具目标对象未实现此方法 if(!this.equalsDefined && AopUtils.isEqualsMethod(method)) {            Boolean retVal2 = Boolean.valueOf(this.equals(args[0]));            return retVal2;} //hashCode()方法,具目标对象未实现此方法  if(!this.hashCodeDefined && AopUtils.isHashCodeMethod(method)) {            Integer retVal1 = Integer.valueOf(this.hashCode());            return retVal1;} //Advised接口或者其父接口中定义的方法,直接反射调用,不应用通知  if(this.advised.opaque || !method.getDeclaringClass().isInterface() || !method.getDeclaringClass().isAssignableFrom(Advised.class)) {if(this.advised.exposeProxy) {                oldProxy = AopContext.setCurrentProxy(proxy);setProxyContext = true;} //获得目标对象的类              target = targetSource.getTarget();            if(target != null) {                targetClass = target.getClass();} //获取可以应用到此方法上的Interceptor列表   <strong> List chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, targetClass);</strong>            if(chain.isEmpty()) {   //如果没有可以应用到此方法的通知(Interceptor),此直接反射调用 method.invoke(target, args)                 retVal = AopUtils.invokeJoinpointUsingReflection(target, method, args);} else {           //创建MethodInvocation ,封装 Interceptor调用链 ReflectiveMethodInvocation invocation = new ReflectiveMethodInvocation(proxy, target, method, args, targetClass, chain);         //调用proceed()方法, 依次执行 Interceptor调用链<strong>retVal = invocation.proceed();</strong>}            Class returnType = method.getReturnType();            if(retVal != null && retVal == target && returnType.isInstance(proxy) && !RawTargetAccess.class.isAssignableFrom(method.getDeclaringClass())) {                retVal = proxy;} else if(retVal == null && returnType != Void.TYPE && returnType.isPrimitive()) {throw new AopInvocationException("Null return value from advice does not match primitive return type for: " + method);}            Object var13 = retVal;            return var13;}        retVal = AopUtils.invokeJoinpointUsingReflection(this.advised, method, args);} finally {if(target != null && !targetSource.isStatic()) {            targetSource.releaseTarget(target);}if(setProxyContext) {            AopContext.setCurrentProxy(oldProxy);}    }return retVal;}
通过this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, targetClass)方法,获取通知拦截器链 
public List<Object> getInterceptorsAndDynamicInterceptionAdvice(Method method, Class<?> targetClass) {//这里使用了cache,利用cache获取已经有了的intercepterAdvisedSupport.MethodCacheKey cacheKey = new AdvisedSupport.MethodCacheKey(method);List cached = (List)this.methodCache.get(cacheKey);if(cached == null) {//这里是利用DefaultAdvisorChainFactory,获取intercepter链cached = this.advisorChainFactory.getInterceptorsAndDynamicInterceptionAdvice(this, method, targetClass);this.methodCache.put(cacheKey, cached);}    return cached;}
DefaultAdvisorChainFactory类(生成连接器的工厂)中:
public List<Object> getInterceptorsAndDynamicInterceptionAdvice(Advised config, Method method, Class<?> targetClass) {    ArrayList interceptorList = new ArrayList(config.getAdvisors().length);    Class actualClass = targetClass != null?targetClass:method.getDeclaringClass();    boolean hasIntroductions = hasMatchingIntroductions(config, actualClass);    AdvisorAdapterRegistry registry = GlobalAdvisorAdapterRegistry.getInstance();    Advisor[] var8 = config.getAdvisors();    int var9 = var8.length;    for(int var10 = 0; var10 < var9; ++var10) {        Advisor advisor = var8[var10];        MethodInterceptor[] interceptors1;        if(advisor instanceof PointcutAdvisor) {            PointcutAdvisor interceptors = (PointcutAdvisor)advisor;            if(config.isPreFiltered() || interceptors.getPointcut().getClassFilter().matches(actualClass)) {                interceptors1 = registry.getInterceptors(advisor);                MethodMatcher mm = interceptors.getPointcut().getMethodMatcher();                if(MethodMatchers.matches(mm, method, actualClass, hasIntroductions)) {                    if(mm.isRuntime()) {                        MethodInterceptor[] var15 = interceptors1;                        int var16 = interceptors1.length;                        for(int var17 = 0; var17 < var16; ++var17) {                            MethodInterceptor interceptor = var15[var17];                            interceptorList.add(new InterceptorAndDynamicMethodMatcher(interceptor, mm));                        }                    } else {                        interceptorList.addAll(Arrays.asList(interceptors1));                    }                }            }        } else if(advisor instanceof IntroductionAdvisor) {            IntroductionAdvisor var19 = (IntroductionAdvisor)advisor;            if(config.isPreFiltered() || var19.getClassFilter().matches(actualClass)) {                interceptors1 = registry.getInterceptors(advisor);                interceptorList.addAll(Arrays.asList(interceptors1));            }        } else {            MethodInterceptor[] var20 = registry.getInterceptors(advisor);            interceptorList.addAll(Arrays.asList(var20));        }    }    return interceptorList;}
在这个函数中,首先设置一个List,用于保存最后返回的interceptorList, 然后DefaultAdvisorChainFactory会通过一个AdvisorAdapterRegistry 来实现拦截器的注册,将配置中Advice适配包装为interceptor,最终返回。
通过 GlobalAdvisorAdapterRegistry.getInstance()获取
AdvisorAdapterRegistry registry。
GlobalAdvisorAdapterRegistry运用单例模式:其实例是DefaultAdvisorAdapterRegistry类对象:
public abstract class GlobalAdvisorAdapterRegistry {    private static AdvisorAdapterRegistry instance = new DefaultAdvisorAdapterRegistry();    public GlobalAdvisorAdapterRegistry() {    }    public static AdvisorAdapterRegistry getInstance() {        return instance;    }    static void reset() {        instance = new DefaultAdvisorAdapterRegistry();    }}

public class DefaultAdvisorAdapterRegistry implements AdvisorAdapterRegistry, Serializable {    private final List<AdvisorAdapter> adapters = new ArrayList(3);    public DefaultAdvisorAdapterRegistry() {        this.registerAdvisorAdapter(new MethodBeforeAdviceAdapter());        this.registerAdvisorAdapter(new AfterReturningAdviceAdapter());        this.registerAdvisorAdapter(new ThrowsAdviceAdapter());    }    public Advisor wrap(Object adviceObject) throws UnknownAdviceTypeException {        if(adviceObject instanceof Advisor) {            return (Advisor)adviceObject;        } else if(!(adviceObject instanceof Advice)) {            throw new UnknownAdviceTypeException(adviceObject);        } else {            Advice advice = (Advice)adviceObject;            if(advice instanceof MethodInterceptor) {                return new DefaultPointcutAdvisor(advice);            } else {                Iterator var3 = this.adapters.iterator();                AdvisorAdapter adapter;                do {                    if(!var3.hasNext()) {                        throw new UnknownAdviceTypeException(advice);                    }                    adapter = (AdvisorAdapter)var3.next();                } while(!adapter.supportsAdvice(advice));                return new DefaultPointcutAdvisor(advice);            }        }    }    public MethodInterceptor[] getInterceptors(Advisor advisor) throws UnknownAdviceTypeException {        ArrayList interceptors = new ArrayList(3);        Advice advice = advisor.getAdvice();        if(advice instanceof MethodInterceptor) {            interceptors.add((MethodInterceptor)advice);        }        Iterator var4 = this.adapters.iterator();        while(var4.hasNext()) {            AdvisorAdapter adapter = (AdvisorAdapter)var4.next();            if(adapter.supportsAdvice(advice)) {                interceptors.add(adapter.getInterceptor(advisor));            }        }        if(interceptors.isEmpty()) {            throw new UnknownAdviceTypeException(advisor.getAdvice());        } else {            return (MethodInterceptor[])interceptors.toArray(new MethodInterceptor[interceptors.size()]);        }    }    public void registerAdvisorAdapter(AdvisorAdapter adapter) {        this.adapters.add(adapter);    }}













0 0
原创粉丝点击