上一篇《學習AOP之認識一下SpringAOP》 中大致的瞭解了代理、動態代理及SpringAop的知識。由於寫的篇幅長了點因此仍是再寫一篇吧。接下來開始深刻一點Spring aop的一些實現機制。javascript
上篇中最後有那段代碼使用了一個ProxyFactory類來完成代理的工做,從而實現了Aop的Around Advice,代碼以下:html
package aop.demo;
import org.springframework.aop.framework.ProxyFactory;
public class ClientCode {
public static void main(String[] args) {
ProxyFactory proxyFactory = new ProxyFactory(); // 建立代理工廠
proxyFactory.setTarget(new SayImpl()); // 射入目標類對象
proxyFactory.addAdvice(new SayImplAroundAdvice());
ISay say = (ISay) proxyFactory.getProxy();
say.say();
}
}
那麼接下來就聊聊ProxyFactory吧,看看它都幹了些啥。java
一、ProxyFactory的奧祕
繼續看上面的代碼只用了5行,這裏面意思也很是明確,只有在第4行的時候有一個getProxy的方法並轉換爲ISay接口。看來代理對象的來源能夠從它入手了。正則表達式
public Object getProxy() {
return createAopProxy().getProxy();
}
只不過代碼只有一行,調用的是一個createAopProxy()的方法返回了AopProxy類型的對象,再經過AopProxy的getProxy來得到了代理對象。spring
那麼只好再看一下createAopProxy()是啥樣子咯:markdown
protected final synchronized AopProxy createAopProxy() {
if (!this.active) {
activate();
}
return getAopProxyFactory().createAopProxy(this);
}
這個方法在org.springframework.aop.framework.ProxyCreatorSupport這個類裏面,ProxyFactory是繼承了它的。這個類字面意思就是一個代理建立的支持類。async
可是看了createAopProxy方法後又鬱悶了,還有一個getAopProxyFactory(),真是一層套一層啊。固然這裏仍是須要從類的層次結構來看會清楚一些,只是我主要是看它是怎麼生成代理對象的,設計上的事情回頭再看。ide
//這個方法訪問了一個內部成員
public AopProxyFactory getAopProxyFactory() {
return this.aopProxyFactory;
}
//再看aopProxyFactory實際上是在構造函數裏建立的
public ProxyCreatorSupport() {
this.aopProxyFactory = new DefaultAopProxyFactory();
}
這裏看到了DefaultAopProxyFactory這個工廠類,好了,也就是它纔是建立代理的真正人物。那麼這裏接着createAopProxy直接看代碼:函數
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()) {
return new JdkDynamicAopProxy(config);
}
return CglibProxyFactory.createCglibProxy(config);
}
else {
return new JdkDynamicAopProxy(config);
}
}
從這裏能夠看到有兩種AopProxy的代理:Cglib和Jdk。它們倆的區別:post
Cglib建立代理慢但執行快並且能夠代理類
Jdk建立代理快但執行慢,只能夠代理接口
順着ClientCode這個代碼確定是用JdkDynmaicAopProxy,最終proxyFactory.getProxy()調用的是JdkDynmaicAopProxy的實例。那好就看一下JdkDynmaicAopProxy中getProxy都作了啥吧:
public Object getProxy() {
return getProxy(ClassUtils.getDefaultClassLoader());
}
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);
findDefinedEqualsAndHashCodeMethods(proxiedInterfaces);
return Proxy.newProxyInstance(classLoader, proxiedInterfaces, this);
}
好了,這裏看到了熟悉的代碼,即經過Proxy.newProxyInstance生成代理對象交給調用者。Spring經過抽象工廠模式設計了兩種代理方法。
二、再看看ProxyFactroyBean
可是在xml配置的時候用的並非ProxyFactory,而是ProxyFactroyBean。有點奇怪,爲何會有兩個類呢?先來看看ProxyFactoryBean:
public class ProxyFactoryBean extends ProxyCreatorSupport
implements FactoryBean<Object>, BeanClassLoaderAware, BeanFactoryAware {
哦喲,原來這傢伙繼承了FactoryBean,好了,原來它借且FactoryBean提供了一箇中間層。由於Spring若是發現Ioc建立的對象帶有FactoryBean時會調用FactoryBean的getObject方法來得到對象。有了這個它在Ioc容器裏得到的即是getObject返回的代理對象,而不是返回ProxyFactoryBean自己,這樣才能注入嘛。因此ProxyFactoryBean主要是使用在Ioc容器裏的。
具體getObject裏實現的原理和ProxyFactory相似,主要仍是和ProxyCreatorSupport有關,ProxyCreatorSupport封裝了這部分邏輯,因此能夠複用。
三、進入切面的小世界 寫了這麼多發現我仍是停留在「代理」的層面,可是AOP難道僅僅止於此嗎?固然不是,好比ISay接口增長一個noaop方法,這個方法我就不但願被代理,那怎麼作呢?
先調整一下例子代碼,增長noaop方法。
public interface ISay {
void say();
void noaop();
}
public class SayImpl implements ISay{
public void say() {
System.out.print("我是5207.");
}
public void noaop() {
System.out.println("別aop我");
}
} 好了,而後增長一個切面,讓這個切面去作分辨,以xml配置爲例,下面對spring.xml作一下修改:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd">
<!-- 聲明被代理的目標對象 -->
<bean id="sayImpl" class="aop.demo.SayImpl"></bean>
<!-- 聲明用於加強的攔截器對象 -->
<bean id="sayImplAroundAdvice" class="aop.demo.SayImplAroundAdvice"></bean>
<!-- 配置一個切面 -->
<bean id="sayAdvisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
<property name="advice" ref="sayImplAroundAdvice"/> <!-- 加強 -->
<property name="pattern" value="aop.demo.SayImpl.s.*"/> <!-- 切點(正則表達式) -->
</bean>
<!-- 聲明代理對象 -->
<bean id="sayProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
<property name="interfaces" value="aop.demo.ISay"/> <!-- 這個就是被代理的接口 -->
<property name="target" ref="sayImpl"/> <!-- 這個就是被代理的對象 -->
<property name="interceptorNames" value="sayAdvisor"/> <!-- 這個就是代理的加強器 -->
</bean>
</beans> 上面xml中新增了一個切面sayAdvisor,它的做用是以正則表達式的規則來選擇是否aop。好比本例子的意思是隻代理SyaImpl的s開頭的方法。那麼noaop方法應該是不會被代理啦。
client代碼也修改一下:
public class Client {
@SuppressWarnings("resource")
public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext("aop/demo/spring.xml");
ISay say = (ISay)context.getBean("sayProxy");
say.say();
say.noaop();//增長noaop的調用,看看會不會被代理
}
} 執行的結果以下:
你們好:我是5207.但願你們多多點贊. 別aop我
這就發現advisor已經有效果啦。
四、自動完成對切面的代理 以前的各類代碼都帶有一個問題,就是client最終調用的時候都是得到的代理對象,以下面的代碼:
ISay say = (ISay)context.getBean("sayProxy"); 那在作Aop加強的時候改老的代碼,這樣就失敗了Aop的意義了。因此沒有辦法能夠自動就完成這個操做,只要配置好就能夠透明的完成這個代理過程呢?
spring提供了自動代理的實現,對spring.xml作一下調整:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd">
<!-- 聲明被代理的目標對象 -->
<bean id="sayImpl" class="aop.demo.SayImpl"></bean>
<!-- 聲明用於加強的攔截器對象 -->
<bean id="sayImplAroundAdvice" class="aop.demo.SayImplAroundAdvice"></bean>
<!-- 配置一個切面 -->
<bean id="sayAdvisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
<property name="advice" ref="sayImplAroundAdvice"/> <!-- 加強 -->
<property name="pattern" value="aop.demo.SayImpl.s.*"/> <!-- 切點(正則表達式) -->
</bean>
<bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator">
<property name="optimize" value="true"/>
</bean>
</beans> 在此增長一個DefaultAdvisorAutoProxyCreator,原先的代理就不須要啦。而後再看一下客戶端調用直接改爲調用SayImpl,看看能不能實現代理:
package aop.demo;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Client {
@SuppressWarnings("resource")
public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext("aop/demo/spring.xml");
ISay say = (ISay)context.getBean("sayImpl");
say.say();
say.noaop();
}
} 輸出:
你們好:我是5207.但願你們多多點贊. 別aop我
效果達成,只不過這裏的關鍵是DefaultAdvisorAutoProxyCreater是怎麼作的呢?看了看代碼發現其主要是藉助Ioc容器在初始化對象時完成的代理的自動生成的。在BeanPostProccer的postProcessAfterInitialization過程當中完成了對代理的生成。具體的原理能夠參考引用中的文章,太累了不寫了。
參考及引用 死磕Spring AOP系列2:剖析Bean處理器之BeanNameAutoProxyCreator
注:此文章爲原創,歡迎轉載,請在文章頁面明顯位置給出此文連接! 若您以爲這篇文章還不錯請點擊下右下角的推薦,很是感謝! http://www.cnblogs.com/5207
posted @
2016-11-11 17:52
5207 閱讀(
... ) 評論(
)
編輯
收藏