我们知道动态代理有两种,分别是基于JDK的动态代理和基于Cglib的动态代理,这两种在Spring AOP中都有使用到,那么两种分别在什么时候用到了呢?这就要看一下源码了,代码片段如下
//org.springframework.aop.framework.DefaultAopProxyFactory
//参数AdvisedSupport 是Spring AOP配置相关类
@Override
public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) {
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.");
}
if (targetClass.isInterface() || Proxy.isProxyClass(targetClass)) {
return new JdkDynamicAopProxy(config);
}
return new ObjenesisCglibAopProxy(config);
}
else {
return new JdkDynamicAopProxy(config);
}
}
从上述源码片段可以看出,是否使用cglib是在代码中进行判断的,判断条件是config.isOptimize()、config.isProxyTargetClass()和hasNoUserSuppliedProxyInterfaces(config)。其中config.isOptimize()与config.isProxyTargetClass()默认返回都是false,这种情况下判断结果就由hasNoUserSuppliedProxyInterfaces(config)的结果决定,该方法实现如下
private boolean hasNoUserSuppliedProxyInterfaces(AdvisedSupport config) {
Class<?>[] ifcs = config.getProxiedInterfaces();
return (ifcs.length == 0 || (ifcs.length == 1 && SpringProxy.class.isAssignableFrom(ifcs[0])));
}
@Override
public Class<?>[] getProxiedInterfaces() {
return this.interfaces.toArray(new Class<?>[this.interfaces.size()]);
}
简单来说hasNoUserSuppliedProxyInterfaces(config)就是在判断代理的对象是否有实现接口,有实现接口的话直接走new JdkDynamicAopProxy(config)分支,即使用JDK的动态代理。
此处还用到了isAssignableFrom方法,其实现如下:
public native boolean isAssignableFrom(Class<?> cls)
可以看出它是一个本地方法,于是查阅了一些资料,官方原文如下
Determines if the class or interface represented by this Class object is
either the same as, or is a superclass or superinterface of, the class or
interface represented by the specified Class parameter. It returns true
if so; otherwise it returns false. If this Class object represents a
primitive type, this method returns true if the specified Class parameter
is exactly this Class object; otherwise it returns false.
Specifically, this method tests whether the type represented by the
specified Class parameter can be converted to the type represented by
this Class object via an identity conversion or via a widening reference
conversion. See The Java Language Specification, sections 5.1.1 and 5.1.4
, for details.
Parameters:
cls the Class object to be checked
Returns:
the boolean value indicating whether objects of the type cls can be
assigned to objects of this class
Throws:
NullPointerException if the specified Class parameter is null
Since:
JDK 1.1
可参考的翻译为
判定一个当前Class对象表示的类或者接口与传入的Class参数相比,是否相同,或者
是否是它的一个超类或者超接口。是的话返回true,否则就返回false。当这个Class
对象表示一种原始类型时,如果传入的Class对象参数就是这个Class对象,那么返回
真,否则为假。
特别的,这个方法可以检查传入的Class参数所代表的类是否可以通过一个类型转换,
或者一个扩大的引用转换转化为当前Class对象所表示的类型。更多细节请参考Java语
言规范 5.1.1 和 5.1.4 两部分章节。
参数: cls 需要检查的Class对象
返回值:一个表示cls类的对象是否可以转换为这个类对象的布尔值
抛出的异常:如果传入的cls参数为空,则抛出空指针异常
版本:JDK 1.1
总结起来就是说isAssignableFrom方法用于判断两个类或者接口是否是相同的类,或者入参类是否是该类的子类。
同时可以发现一个问题,即config.isOptimize()或config.isProxyTargetClass()为false时可以强制改变运行的分支,让hasNoUserSuppliedProxyInterfaces(config)的判断无效化。
config.isOptimize()和config.isProxyTargetClass()分别由下述配置项进行控制
<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>
<bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator">
<property name="optimize" value="true"/>
</bean>
注:关于getTargetClass这块不是很了解,是在高版本Spring中才增加的东西,留待后面有机会再去深入研究。