JDK和cglib生成代理类
關(guān)于動態(tài)代理和靜態(tài)代理
當(dāng)一個對象(客戶端)不能或者不想直接引用另一個對象(目標(biāo)對象),這時可以應(yīng)用代理模式在這兩者之間構(gòu)建一個橋梁–代理對象。
按照代理對象的創(chuàng)建時期不同,可以分為兩種:
靜態(tài)代理:事先寫好代理對象類,在程序發(fā)布前就已經(jīng)存在了;動態(tài)代理:應(yīng)用程序發(fā)布后,通過動態(tài)創(chuàng)建代理對象。動態(tài)代理又可分為:JDK動態(tài)代理和CGLIB代理。
1.JDK動態(tài)代理
此時代理對象和目標(biāo)對象實現(xiàn)了相同的接口,目標(biāo)對象作為代理對象的一個屬性,具體接口實現(xiàn)中,可以在調(diào)用目標(biāo)對象相應(yīng)方法前后加上其他業(yè)務(wù)處理邏輯。
代理模式在實際使用時需要指定具體的目標(biāo)對象,如果為每個類都添加一個代理類的話,會導(dǎo)致類很多,同時如果不知道具體類的話,怎樣實現(xiàn)代理模式呢?這就引出動態(tài)代理。
JDK動態(tài)代理只能針對實現(xiàn)了接口的類生成代理。
2.CGLIB代理
CGLIB(CODE GENERLIZE LIBRARY)代理是針對類實現(xiàn)代理,
主要是對指定的類生成一個子類,覆蓋其中的所有方法,所以該類或方法不能聲明稱final的。
JDK動態(tài)代理和CGLIB代理生成的區(qū)別
JDK動態(tài)代理只能對實現(xiàn)了接口的類生成代理,而不能針對類 。
CGLIB是針對類實現(xiàn)代理,主要是對指定的類生成一個子類,覆蓋其中的方法 。
因為是繼承,所以該類或方法最好不要聲明成final ,final可以阻止繼承和多態(tài)。
PS:final 所修飾的數(shù)據(jù)具有“終態(tài)”的特征,表示“最終的”意思:
final 修飾的類不能被繼承。 final 修飾的方法不能被子類重寫。 final 修飾的變量(成員變量或局部變量)即成為常量,只能賦值一次。 final 修飾的成員變量必須在聲明的同時賦值,如果在聲明的時候沒有賦值,那么只有 一次賦值的機(jī)會,而且只能在構(gòu)造方法中顯式賦值,然后才能使用。 final 修飾的局部變量可以只聲明不賦值,然后再進(jìn)行一次性的賦值。參考代碼
CGLIB:
public Object createProxyObject(Object obj) { this.targetObject = obj; Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(obj.getClass()); enhancer.setCallback(this); Object proxyObj = enhancer.create(); return proxyObj;// 返回代理對象,返回的對象其實就是一個封裝了“實現(xiàn)類”的代理類,是實現(xiàn)類的實例。 }JDK:
public Object newProxy(Object targetObject) {// 將目標(biāo)對象傳入進(jìn)行代理 this.targetObject = targetObject; <br> //注意這個方法的參數(shù),后面是類實現(xiàn)的接口return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), this);// 返回代理對象 }在代碼中可以看到,在生成代理類時,傳遞的是實現(xiàn)類所實現(xiàn)的接口 targetObject.getClass().getInterfaces(),所以JDK只能對于接口進(jìn)行做代理。如果換成類的話,則會拋java.lang.ClassCastException異常。
在Spring的源碼中,可以看到很多生成代理類的代碼。
動態(tài)代理的應(yīng)用
AOP(Aspect-OrientedProgramming,面向切面編程),AOP包括切面(aspect)、通知(advice)、連接點(diǎn)(joinpoint),實現(xiàn)方式就是通過對目標(biāo)對象的代理在連接點(diǎn)前后加入通知,完成統(tǒng)一的切面操作。
實現(xiàn)AOP的技術(shù),主要分為兩大類:
一是采用動態(tài)代理技術(shù),利用截取消息的方式,對該消息進(jìn)行裝飾,以取代原有對象行為的執(zhí)行;
二是采用靜態(tài)織入的方式,引入特定的語法創(chuàng)建“方面”,從而使得編譯器可以在編譯期間織入有關(guān)“方面”的代碼。
Spring提供了兩種方式來生成代理對象: JDKProxy和Cglib,具體使用哪種方式生成由AopProxyFactory根據(jù)AdvisedSupport對象的配置來決定。
默認(rèn)的策略是如果目標(biāo)類是接口,則使用JDK動態(tài)代理技術(shù),如果目標(biāo)對象沒有實現(xiàn)接口,則默認(rèn)會采用CGLIB代理。
如果目標(biāo)對象實現(xiàn)了接口,可以強(qiáng)制使用CGLIB實現(xiàn)代理(添加CGLIB庫,并在spring配置中加入<aop:aspectj-autoproxy proxy-target-class=“true”/>)。
轉(zhuǎn)自:http://www.cnblogs.com/binyue/p/4519652.html
總結(jié)
以上是生活随笔為你收集整理的JDK和cglib生成代理类的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: JavaScript教程之事件处理
- 下一篇: Java Socket 教程