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 。
分享到:
相关推荐
每个拦截器都是AOP联盟中的`MethodInterceptor`接口的实现,它们会在目标方法的调用前后执行。Spring会按照`interceptors`属性中定义的顺序,依次调用每个拦截器的`invoke()`方法。在这个方法中,拦截器可以执行...
标签“源码”意味着这篇博客可能深入解析了Spring AOP的内部工作机制,包括如何通过代理机制实现切面的织入,以及Spring AOP的相关核心类如`Advised`、`ProxyFactoryBean`、`DefaultAdvisorAdapterRegistry`等。...
2. proxyInterfaces:代理要实现的接口,如果有多个接口,则可以使用以下格式赋值:<list> <value ></value> ... 3. proxyTargetClass:是否对类代理而不是接口,设置为 true 时,使用 CGLIB 代理。 4. ...
Java字节码实现AOP(面向切面编程)是一种在程序运行时动态插入代码的技术,它使得我们可以在不修改原有代码的情况下,增加新的功能或监控已有功能。在Java中,AOP通常通过代理模式和字节码操作来实现,如Spring AOP...
- `AopProxy`根据`ProxyFactoryBean`的配置(如代理类型、拦截器等)生成具体的代理对象,可能是JDK动态代理的`InvocationHandler`实现,也可能是CGLIB的子类。 - 用户通过`ProxyFactoryBean`获取到的bean就是这个...
7. **代理(Proxy)**:应用通知的对象,Spring通常使用JDK动态代理或CGLIB实现。 8. **织入(Weaving)**:将切面应用到目标对象以创建代理对象的过程,可以在编译时、类加载时或运行时进行。 **第一种:经典的基于...
2. CGLIB代理:如果目标类没有实现接口,Spring会使用CGLIB库生成目标类的一个子类,子类覆盖了原类的方法并在方法调用前后插入通知。CGLIB是基于类的代理,它在运行时动态生成字节码,效率相对较低,但能处理没有...
本文将深入解析Spring AOP如何获取代理对象,即Proxy。 首先,我们需要理解AOP的核心概念: 1. **Advice**:通知,是切面实际执行的操作。在Spring中,Advice可以是Before、AfterReturning、AfterThrowing以及...
Spring 实现 AOP 的四种方式 Spring 框架提供了四种方式来实现Aspect-Oriented Programming(AOP),分别是经典的基于代理的 AOP、@AspectJ 注解驱动的切面、纯 POJO 切面和注入式 AspectJ 切面。下面将详细介绍这...
package com.gc.aop下为:aop方式ProxyFactoryBean代理 package com.gc.aop下为:aop方式ProxyFactoryBean代理 package com.gc.autoproxy下为:aop方式自动代理 package com.gc.cglib下为:aop方式cglib代理 ...
虽然CGLIB代理相比JDK动态代理在某些场景下可能稍慢,但它的优势在于无需目标对象实现接口。在实际应用中,应根据项目需求选择合适的代理方式。 8. **AOP的应用场景** AOP在许多场景下都非常有用,如事务管理、...
开发者可以通过实现org.springframework.aop.MethodBeforeAdvice接口来实现Before Advice的逻辑。在before方法中,可以取得指定方法的Method实例、参数列表和目标对象,在before方法执行完后,目标对象上的方法将会...
除了这两个核心接口外,aopalliance还提供了一些辅助类和接口,如`org.aopalliance.aop.Advice`(表示一个通知,即在特定连接点执行的操作),以及`org.aopalliance.aop.Aspect`(表示一个切面,可能包含多个通知)...
这些通知通过`org.springframework.aop.MethodBeforeAdvice`、`org.springframework.aop.AfterReturningAdvice`等接口实现。 三、连接点(Join Point) 连接点是程序执行中的一个特定点,例如方法的调用。在Spring ...
在源码层面,`org.springframework.aop.framework.ProxyFactoryBean`是创建代理的主要类,它会根据目标对象是否实现接口来选择合适的代理方式。如果目标类实现了接口,那么使用JDK动态代理;否则,使用CGLIB代理。`...
在Spring中,AOP主要是通过代理模式实现的,有JDK动态代理和CGLIB两种方式。 对于Spring事务管理,我们通常有两种方式:编程式和声明式。编程式事务管理需要在代码中显式地调用begin、commit、rollback等事务控制...
顾名思义,Before Advice会在目标对象的方法执行之前被调用,您可以通过实现org.springframework.aop.MethodBeforeAdvice接口来实现Before Advice的逻辑,接口定义如下: java 代码 1. package org.spring...
Spring 提供了多种 AOP 支持方式,每种方式都有其特点和适用场景。基于代理的经典 AOP 更适合于 Spring 早期版本的应用场景,而 Schema-based 的 AOP 和 @AspectJ 注解驱动的 AOP 更符合现代应用的需求,特别是在...
如果没有实现接口,Spring则会使用CGLIB代理,通过字节码技术创建一个目标类的子类来实现。 1. **通知(Advises)**: - 前置通知(Before):在目标方法执行前执行。 - 后置通知(After):在目标方法执行后,...