`

ProxyFactoryBean 如何实现AOP proxy都有可能转为Advised接口

阅读更多
1、ProxyFactoryBean 把proxy的创建交给AopProxy去做。

public class ProxyFactoryBean extends AdvisedSupport

    implements FactoryBean, BeanFactoryAware, AdvisedSupportListener {

            public Object getObject() throws BeansException {

                        //invoke AopProxy 去 创建proxy

                        return this.singleton ? getSingletonInstance() : newPrototypeInstance();

            }

           

            //inherited from ProxyConfig

            public void setAopProxyFactory(AopProxyFactory apf) {

                        this.aopProxyFactory = apf;

            }

            //inherited from ProxyConfig

            public AopProxyFactory getAopProxyFactory() {

                        return this.aopProxyFactory;

            }

}



2.、那么AopProxy又如何获取?答案是从AopProxyFactory获得。这个AopProxyFactory已在ProxyFactoryBean掌控范围内,如上述代码所示。AopProxyFactory interface如下所示:



public interface AopProxyFactory {   



            AopProxy createAopProxy(AdvisedSupport advisedSupport) throws AopConfigException;



}

3、AopProxy是如何introduce Advised interface的?那要先从AopProxyFactory的implementation说起。

AopProxyFactory的实现类目前只有DefaultAopProxyFactory。这个类的核心方法如下:

            public AopProxy createAopProxy(AdvisedSupport advisedSupport) throws AopConfigException {

                        if (advisedSupport.isOptimize() || advisedSupport.isProxyTargetClass() ||

                            advisedSupport.getProxiedInterfaces().length == 0) {

                                    if (!cglibAvailable) {

                                                throw new AopConfigException(

                                                                        "Cannot proxy target class because CGLIB2 is not available. " +

                                                                        "Add CGLIB to the class path or specify proxy interfaces.");

                                    }

                                    return CglibProxyFactory.createCglibProxy(advisedSupport);

                      }

                        else {

                                    return new JdkDynamicAopProxy(advisedSupport);

                      }

            }

这个方法主要是依赖CglibProxyFactory 类和JdkDynamicAopProxy类。这两个类是如何代理所有的接口的呢?

这两个类都是重要接口AopProxy的实现类,这个接口的核心方法是

            public Object getProxy(ClassLoader classLoader)。

其中JdkDynamicAopProxy实现该方法用到的获取需要代理的接口的代码如下:

            public Object getProxy(ClassLoader classLoader) {

                        if (logger.isDebugEnabled()) {

                                    Class targetClass = this.advised.getTargetSource().getTargetClass();

                                    logger.debug("Creating JDK dynamic proxy" +

                                                            (targetClass != null ? " for [" + targetClass.getName() + "]" : ""));

                      }

                        Class[] proxiedInterfaces = AopProxyUtils.completeProxiedInterfaces(this.advised);

                        return Proxy.newProxyInstance(classLoader, proxiedInterfaces, this);

            }



而CglibProxyFactory的getProxy(ClassLoader classLoader)方法如下:

            public Object getProxy(ClassLoader classLoader) {

……

                        Enhancer enhancer = new Enhancer();                       

……

            enhancer.setInterfaces(AopProxyUtils.completeProxiedInterfaces(this.advised));

……

}



从上面可以看出,这两个AopProxy实现方案在获取接口时都是通过AopProxyUtils.completeProxiedInterfaces(this.advised)获取的。该方法如下:

            /**

             * Get complete set of interfaces to proxy. This will always add the Advised interface

             * unless the AdvisedSupport's "opaque" flag is true.

             * @return the complete set of interfaces to proxy

             */

            public static Class[] completeProxiedInterfaces(AdvisedSupport advised) {

                        // Won't include Advised, which may be necessary.

                        Class[] specifiedInterfaces = advised.getProxiedInterfaces();

                        Class[] proxiedInterfaces = specifiedInterfaces;

                        if (!advised.isOpaque() && !advised.isInterfaceProxied(Advised.class)) {

                                    // We need to add the Advised interface.

                                    proxiedInterfaces = new Class[specifiedInterfaces.length + 1];

                                    proxiedInterfaces[0] = Advised.class;

                                    System.arraycopy(specifiedInterfaces, 0, proxiedInterfaces, 1, specifiedInterfaces.length);

                      }

                        return proxiedInterfaces;

            }

从这可以看出,是在这强制加了对Advised  interface的实现。前提是是否允许代理该interafce,这由isOpaque方法确定,该方法Return whether proxies created by this configuration should be prevented from being cast to Advised.



4. 顺便提一下,Advised interface的具体实现又是怎样的,在哪呢?

4.1 在JdkDynamicAopProxy里,



            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

……

                                    if (Advised.class == method.getDeclaringClass()) {

                                                // service invocations on ProxyConfig with the proxy config

                                                return AopUtils.invokeJoinpointUsingReflection(this.advised, method, args);

                                    }

……

}

注意第一个参数是this.advised,其实就是ProxyFactoryBean本身,注意在我发表的其天的Spring文章里也曾提到,ProxyFactoryBean extends AdvisedSupport,

而AdisedSupport extends ProxyConfig implements Advised,也就是说Advised的默认实现就是AdisedSupport。



4.2 在Cglib2AopProxy里(如果不熟Cglib,建议先看看其天有关的Cglib资料),





            public Object getProxy(ClassLoader classLoader) {

……

Enhancer enhancer = new Enhancer();

……

enhancer.setCallbackFilter(new ProxyCallbackFilter(this.advised));

……

Callback[] callbacks = getCallbacks(rootClass);

……

……

……

}

private Callback[] getCallbacks(Class rootClass) throws Exception {

……

                        Callback[] mainCallbacks = new Callback[]{

                                    aopInterceptor, // for normal advice

                                    targetInterceptor, // invoke target without considering advice, if optimized

                                    new SerializableNoOp(), // no override for methods mapped to this

                                    targetDispatcher, this.advisedDispatcher,

                                    new EqualsInterceptor(this.advised)

                        };

……

}



private class ProxyCallbackFilter implements CallbackFilter {

public int accept(Method method) {

                                    if (method.getDeclaringClass() == Advised.class) {

                                                if (logger.isDebugEnabled()) {

                                                            logger.debug("Method " + method + " is declared on Advised - using DISPATCH_ADVISED");

                                              }

                                                return DISPATCH_ADVISED;// DISPATCH_ADVISED 值为4.

                                    }

……

}

}



private final transient AdvisedDispatcher advisedDispatcher = new AdvisedDispatcher();



            /**

             * Dispatcher for any methods declared on the Advised class.

             */

            private class AdvisedDispatcher implements Dispatcher, Serializable {



                        public Object loadObject() throws Exception {

                                    return advised;

                      }

            }



注意Dispatcher extends Callback interface 。


分享到:
评论

相关推荐

    Spring AOP 源码系列:ProxyFactoryBean

    每个拦截器都是AOP联盟中的`MethodInterceptor`接口的实现,它们会在目标方法的调用前后执行。Spring会按照`interceptors`属性中定义的顺序,依次调用每个拦截器的`invoke()`方法。在这个方法中,拦截器可以执行...

    spring-aop AND proxy

    标签“源码”意味着这篇博客可能深入解析了Spring AOP的内部工作机制,包括如何通过代理机制实现切面的织入,以及Spring AOP的相关核心类如`Advised`、`ProxyFactoryBean`、`DefaultAdvisorAdapterRegistry`等。...

    Spring基于ProxyFactoryBean创建AOP代理

    2. proxyInterfaces:代理要实现的接口,如果有多个接口,则可以使用以下格式赋值:<list> <value ></value> ... 3. proxyTargetClass:是否对类代理而不是接口,设置为 true 时,使用 CGLIB 代理。 4. ...

    Java字节码实现Aop

    Java字节码实现AOP(面向切面编程)是一种在程序运行时动态插入代码的技术,它使得我们可以在不修改原有代码的情况下,增加新的功能或监控已有功能。在Java中,AOP通常通过代理模式和字节码操作来实现,如Spring AOP...

    spring-aop-ProxyFactoryBean 源码分析

    - `AopProxy`根据`ProxyFactoryBean`的配置(如代理类型、拦截器等)生成具体的代理对象,可能是JDK动态代理的`InvocationHandler`实现,也可能是CGLIB的子类。 - 用户通过`ProxyFactoryBean`获取到的bean就是这个...

    Spring实现AOP的四种方式

    7. **代理(Proxy)**:应用通知的对象,Spring通常使用JDK动态代理或CGLIB实现。 8. **织入(Weaving)**:将切面应用到目标对象以创建代理对象的过程,可以在编译时、类加载时或运行时进行。 **第一种:经典的基于...

    spring源代码分析:aop的实现

    2. CGLIB代理:如果目标类没有实现接口,Spring会使用CGLIB库生成目标类的一个子类,子类覆盖了原类的方法并在方法调用前后插入通知。CGLIB是基于类的代理,它在运行时动态生成字节码,效率相对较低,但能处理没有...

    Spring源代码解析(五):Spring_AOP获取Proxy.doc

    本文将深入解析Spring AOP如何获取代理对象,即Proxy。 首先,我们需要理解AOP的核心概念: 1. **Advice**:通知,是切面实际执行的操作。在Spring中,Advice可以是Before、AfterReturning、AfterThrowing以及...

    Spring实现AOP的四种方式.pdf

    Spring 实现 AOP 的四种方式 Spring 框架提供了四种方式来实现Aspect-Oriented Programming(AOP),分别是经典的基于代理的 AOP、@AspectJ 注解驱动的切面、纯 POJO 切面和注入式 AspectJ 切面。下面将详细介绍这...

    spring aop代理列子

    package com.gc.aop下为:aop方式ProxyFactoryBean代理 package com.gc.aop下为:aop方式ProxyFactoryBean代理 package com.gc.autoproxy下为:aop方式自动代理 package com.gc.cglib下为:aop方式cglib代理 ...

    Spring AOP的底层实现技术

    虽然CGLIB代理相比JDK动态代理在某些场景下可能稍慢,但它的优势在于无需目标对象实现接口。在实际应用中,应根据项目需求选择合适的代理方式。 8. **AOP的应用场景** AOP在许多场景下都非常有用,如事务管理、...

    个人对spring实现AOP的方法总结

    开发者可以通过实现org.springframework.aop.MethodBeforeAdvice接口来实现Before Advice的逻辑。在before方法中,可以取得指定方法的Method实例、参数列表和目标对象,在before方法执行完后,目标对象上的方法将会...

    aopalliance源码

    除了这两个核心接口外,aopalliance还提供了一些辅助类和接口,如`org.aopalliance.aop.Advice`(表示一个通知,即在特定连接点执行的操作),以及`org.aopalliance.aop.Aspect`(表示一个切面,可能包含多个通知)...

    spring-aop源码

    这些通知通过`org.springframework.aop.MethodBeforeAdvice`、`org.springframework.aop.AfterReturningAdvice`等接口实现。 三、连接点(Join Point) 连接点是程序执行中的一个特定点,例如方法的调用。在Spring ...

    spring aop 源码解析

    在源码层面,`org.springframework.aop.framework.ProxyFactoryBean`是创建代理的主要类,它会根据目标对象是否实现接口来选择合适的代理方式。如果目标类实现了接口,那么使用JDK动态代理;否则,使用CGLIB代理。`...

    spring基于AOP实现事务

    在Spring中,AOP主要是通过代理模式实现的,有JDK动态代理和CGLIB两种方式。 对于Spring事务管理,我们通常有两种方式:编程式和声明式。编程式事务管理需要在代码中显式地调用begin、commit、rollback等事务控制...

    spring aop 实现源代码--xml and annotation(带lib包)

    顾名思义,Before Advice会在目标对象的方法执行之前被调用,您可以通过实现org.springframework.aop.MethodBeforeAdvice接口来实现Before Advice的逻辑,接口定义如下: java 代码 1. package org.spring...

    spring入门学习-6、AOP几种配置方式详解.pdf

    Spring 提供了多种 AOP 支持方式,每种方式都有其特点和适用场景。基于代理的经典 AOP 更适合于 Spring 早期版本的应用场景,而 Schema-based 的 AOP 和 @AspectJ 注解驱动的 AOP 更符合现代应用的需求,特别是在...

    五、Spring源码分析——Spring Aop

    如果没有实现接口,Spring则会使用CGLIB代理,通过字节码技术创建一个目标类的子类来实现。 1. **通知(Advises)**: - 前置通知(Before):在目标方法执行前执行。 - 后置通知(After):在目标方法执行后,...

    spring2-aop.pdf

    - **Spring AOP 实现**:Spring 使用代理模式实现 AOP 功能,支持基于接口的动态代理和 CGLIB 字节码生成库。 - **Spring AOP 对 AspectJ 的支持**:Spring 允许使用 AspectJ 的注解来定义切面和切点。 - **Spring ...

Global site tag (gtag.js) - Google Analytics