Spring之AOP兩種代理機(jī)制對比分析(JDK和CGLib動態(tài)代理)
Spring AOP兩種代理機(jī)制對比
Spirng的AOP的動態(tài)代理實(shí)現(xiàn)機(jī)制有兩種,分別是:
JDK動態(tài)代理
具體實(shí)現(xiàn)原理:
1、通過實(shí)現(xiàn)InvocationHandlet接口創(chuàng)建自己的調(diào)用處理器
2、通過為Proxy類指定ClassLoader對象和一組interface來創(chuàng)建動態(tài)代理
3、通過反射機(jī)制獲取動態(tài)代理類的構(gòu)造函數(shù),其唯一參數(shù)類型就是調(diào)用處理器接口類型
4、通過構(gòu)造函數(shù)創(chuàng)建動態(tài)代理類實(shí)例,構(gòu)造時(shí)調(diào)用處理器對象作為參數(shù)參入
JDK動態(tài)代理是面向接口的代理模式,如果被代理目標(biāo)沒有接口那么Spring也無能為力,
Spring通過java的反射機(jī)制生產(chǎn)被代理接口的新的匿名實(shí)現(xiàn)類,重寫了其中AOP的增強(qiáng)方法。
CGLib動態(tài)代理
CGLib是一個(gè)強(qiáng)大、高性能的Code生產(chǎn)類庫,可以實(shí)現(xiàn)運(yùn)行期動態(tài)擴(kuò)展java類,Spring在運(yùn)行期間通過 CGlib繼承要被動態(tài)代理的類,重寫父類的方法,實(shí)現(xiàn)AOP面向切面編程呢。
兩者對比:
JDK動態(tài)代理是面向接口,在創(chuàng)建代理實(shí)現(xiàn)類時(shí)比CGLib要快,創(chuàng)建代理速度快。
CGLib動態(tài)代理是通過字節(jié)碼底層繼承要代理類來實(shí)現(xiàn)(如果被代理類被final關(guān)鍵字所修飾,那么抱歉會失?。?,在創(chuàng)建代理這一塊沒有JDK動態(tài)代理快,但是運(yùn)行速度比JDK動態(tài)代理要快。
使用注意:
如果要被代理的對象是個(gè)實(shí)現(xiàn)類,那么Spring會使用JDK動態(tài)代理來完成操作(Spirng默認(rèn)采用JDK動態(tài)代理實(shí)現(xiàn)機(jī)制)
如果要被代理的對象不是個(gè)實(shí)現(xiàn)類那么,Spring會強(qiáng)制使用CGLib來實(shí)現(xiàn)動態(tài)代理。
那么如何選擇的使用代理機(jī)制了?

通過配置Spring的中<aop:config>標(biāo)簽來顯示的指定使用動態(tài)代理機(jī)制 proxy-target-class=true表示使用CGLib代理,如果為false就是默認(rèn)使用JDK動態(tài)代理
SpringAOP兩種代理原理
SpringAOP代理
spingAOP代理有兩種:
- JDK動態(tài)代理:目標(biāo)類必須實(shí)現(xiàn)一個(gè)接口
- CGLIB代理:目標(biāo)類必須繼承一個(gè)類
JDK動態(tài)代理
JDK為什么一定要目標(biāo)類實(shí)現(xiàn)一個(gè)接口呢,這其實(shí)就得看看JDK動態(tài)代理的原理了,其實(shí)JDK動態(tài)代理它是先生成一個(gè)代理類然后他也是實(shí)現(xiàn)了目標(biāo)類實(shí)現(xiàn)的接口里面的方法,只是他還是調(diào)用的是目標(biāo)類的方法。
下面我們來自定義實(shí)現(xiàn)一下
//創(chuàng)建一個(gè)接口
public interface StudentBiz {
int add(String name);
void update(String name);
void find (String name);
}//創(chuàng)建一個(gè)類實(shí)現(xiàn)那個(gè)接口
public class StudentBizimpl implements StudentBiz{
@Override
public int add(String name) {
System.out.println("調(diào)用了studentBizimpl中的add"+name);
return 100;
}
@Override
public void update(String name) {
System.out.println("調(diào)用了studentBizimpl中的update"+name);
}
@Override
public void find(String name) {
System.out.println("調(diào)用了studentBizimpl中的find"+name);
}
}//jdk動態(tài)代理三大重點(diǎn)
// 1.有目標(biāo)類的引用
// 2.有一個(gè)創(chuàng)建代理實(shí)例的方法createProxy()里面有Proxy.newProxyInstance()方法來創(chuàng)建代理實(shí)例
// 3.有一個(gè)回調(diào)方法invoke
public class LogAspect implements InvocationHandler {
private Object target;//目標(biāo)類的對象
public LogAspect(Object target){
this.target=target;
}
public Object createProxy(){
//新建一個(gè)代理實(shí)例
// 第一個(gè)參數(shù)是類加載器 第二個(gè)是得獲取到目標(biāo)類實(shí)現(xiàn)的接口 第三個(gè)是代理類對象
return Proxy.newProxyInstance(this.target.getClass().getClassLoader(),this.target.getClass().getInterfaces(),this);
}
@Override//回調(diào)方法 當(dāng)jvm調(diào)用代理對象的被代理的方法時(shí),會由jvm自動調(diào)用這個(gè)invoke
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("代理類對象:"+proxy.getClass());
System.out.println("目標(biāo)的方法:"+method);
System.out.println("方法的參數(shù):"+args);
log();//這里就可以加增強(qiáng) 具體哪些方法加得看切入點(diǎn)表達(dá)式來判斷
Object o=method.invoke(this.target,args);//激活目標(biāo)類方法
return o;
}
private void log(){
System.out.println("前置增強(qiáng)");
}
}下面我們再來做一個(gè)測試類:
public class Test {
public static void main(String[] args) throws IllegalAccessException, InstantiationException {
StudentBiz sbm=new StudentBizimpl();
LogAspect la=new LogAspect(sbm);//放目標(biāo)類對象
Object o=la.createProxy();//創(chuàng)建一個(gè)代理類
if (o instanceof StudentBiz){
StudentBiz sb=(StudentBiz)o;
sb.add("張三");//這里就會自動回調(diào)invoke方法
}
}
}得到以下結(jié)果:

CGLIB代理
CGLib是一個(gè)強(qiáng)大、高性能的Code生產(chǎn)類庫,可以實(shí)現(xiàn)運(yùn)行期動態(tài)擴(kuò)展java類,Spring在運(yùn)行期間通過 CGlib繼承要被動態(tài)代理的類,重寫父類的方法,實(shí)現(xiàn)AOP面向切面編程呢。
CGLIB代理其實(shí)也就是生成一個(gè)代理對象他也繼承了目標(biāo)類的父類中的方法,再通過回調(diào)自身引用目標(biāo)類的方法完成代理.
下面來簡單地自定義實(shí)現(xiàn)一下
//做一個(gè)cglib代理類
public class LogAspectcglib implements MethodInterceptor {
private Object target;
public LogAspectcglib(Object target){
this.target=target;
}
public Object createProxy(){
Enhancer enhancer=new Enhancer();//用于生成代理對象
enhancer.setSuperclass(this.target.getClass());//設(shè)置父類
enhancer.setCallback(this);//設(shè)置回調(diào)用對象為本身
return enhancer.create();//創(chuàng)建代理對象
}
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("代理類對象"+o.getClass());
System.out.println("目標(biāo)類的方法"+method);
System.out.println("目標(biāo)方法參數(shù)"+objects);
System.out.println("要代理的方法"+methodProxy);
Object returnvalue= method.invoke(this.target,objects);
return returnvalue;
}
}測試類:
public class Test {
public static void main(String[] args) throws IllegalAccessException, InstantiationException {
StudentBizimpl sbm=new StudentBizimpl();
LogAspectcglib lac=new LogAspectcglib(sbm);
Object o=lac.createProxy();
if (o instanceof StudentBizimpl){
StudentBizimpl sb=(StudentBizimpl) o;
sb.add("張三");
}
}
}得到結(jié)果:

兩者對比
- JDK動態(tài)代理是面向接口,在創(chuàng)建代理實(shí)現(xiàn)類時(shí)比CGLib要快,創(chuàng)建代理速度快。
- CGLib動態(tài)代理是通過字節(jié)碼底層繼承要代理類來實(shí)現(xiàn)(如果被代理類被final關(guān)鍵字所修飾,那么抱歉會失?。?,在創(chuàng)建代理這一塊沒有JDK動態(tài)代理快,但是運(yùn)行速度比JDK動態(tài)代理要快。
使用注意
如果要被代理的對象是個(gè)實(shí)現(xiàn)類,那么Spring會使用JDK動態(tài)代理來完成操作(Spirng默認(rèn)采用JDK動態(tài)代理實(shí)現(xiàn)機(jī)制)
如果要被代理的對象不是個(gè)實(shí)現(xiàn)類那么,Spring會強(qiáng)制使用CGLib來實(shí)現(xiàn)動態(tài)代理。
如果要強(qiáng)制使用CGLIB代理則需在xml中配置如下:
<aop:aspectj-autoproxy proxy-target-class="true"/>
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- 深入了解SpringAOP中的jdk動態(tài)代理與CGlib
- Java兩種動態(tài)代理JDK動態(tài)代理和CGLIB動態(tài)代理詳解
- Java的動態(tài)代理模式之JDK代理詳解
- JDK動態(tài)代理提高代碼可維護(hù)性和復(fù)用性利器
- Java JDK與cglib動態(tài)代理有什么區(qū)別
- 解讀jdk動態(tài)代理為什么必須實(shí)現(xiàn)接口
- Java實(shí)現(xiàn)JDK動態(tài)代理的原理詳解
- Java反射(JDK)與動態(tài)代理(CGLIB)詳解
- Java JDK動態(tài)代理在攔截器和聲明式接口中的應(yīng)用小結(jié)
相關(guān)文章
java導(dǎo)出數(shù)據(jù)庫中Excel表格數(shù)據(jù)的方法
這篇文章主要為大家詳細(xì)介紹了java導(dǎo)出數(shù)據(jù)庫中Excel表格數(shù)據(jù)的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-08
java 圖片驗(yàn)證碼的實(shí)現(xiàn)代碼
java 圖片驗(yàn)證碼的實(shí)現(xiàn)代碼,需要的朋友可以參考一下2013-05-05
Java如何自定義類數(shù)組的創(chuàng)建和初始化
這篇文章主要介紹了Java如何自定義類數(shù)組的創(chuàng)建和初始化,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
SpringBoot下載Excel文件時(shí),報(bào)錯(cuò)文件損壞的解決方案
這篇文章主要介紹了SpringBoot下載Excel文件時(shí),報(bào)錯(cuò)文件損壞的解決方案,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06
詳解Java如何通過Socket實(shí)現(xiàn)查詢IP
在本文中,我們來學(xué)習(xí)下如何找到連接到服務(wù)器的客戶端計(jì)算機(jī)的IP地址。我們將創(chuàng)建一個(gè)簡單的客戶端-服務(wù)器場景,讓我們探索用于TCP/IP通信的java.net?API,感興趣的可以了解一下2022-10-10
logback StatusListener的定義方法源碼解讀
這篇文章主要為大家介紹了logback StatusListener的定義方法源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11
Java案例使用比較排序器comparator實(shí)現(xiàn)成績排序
這篇文章主要介紹了Java案例使用比較排序器comparator實(shí)現(xiàn)成績排序,主要通過案例用TreeSet集合存儲多個(gè)學(xué)生信息,并遍歷該集合,要按照總分從高到低進(jìn)行排序,下文介紹需要的朋友可以參考一下2022-04-04
Spring MVC如何使用@RequestParam注解獲取參數(shù)
這篇文章主要介紹了Spring MVC實(shí)現(xiàn)使用@RequestParam注解獲取參數(shù)方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10

