SpringAop實(shí)現(xiàn)原理及代理模式詳解
Spring Aop的原理
Spring的AOP就是通過(guò)動(dòng)態(tài)代理實(shí)現(xiàn)的。當(dāng)為某個(gè)Bean或者某些Bean配置切面時(shí),Spring會(huì)為其創(chuàng)建代理對(duì)象,當(dāng)調(diào)用該對(duì)象的某個(gè)方法時(shí),實(shí)際是調(diào)用生成的代理類的對(duì)象方法。Spring的Aop主要是使用了兩個(gè)動(dòng)態(tài)代理,分別是JDK的動(dòng)態(tài)代理和CGLIB動(dòng)態(tài)代理。
1. JDK動(dòng)態(tài)代理
如果代理類實(shí)現(xiàn)了接口,Spring默認(rèn)會(huì)使用JDK動(dòng)態(tài)代理。JDK的動(dòng)態(tài)代理是基于反射實(shí)現(xiàn)。JDK通過(guò)反射,生成一個(gè)代理類,這個(gè)代理類實(shí)現(xiàn)了原來(lái)那個(gè)類的全部接口,并對(duì)接口中定義的所有方法進(jìn)行了代理。當(dāng)我們通過(guò)代理對(duì)象執(zhí)行原來(lái)那個(gè)類的方法時(shí),代理類底層會(huì)通過(guò)反射機(jī)制,調(diào)用我們實(shí)現(xiàn)的InvocationHandler接口的invoke方法。
/* * 接口類 */ public interface Person { void say(); } * 接口實(shí)現(xiàn)類 public class Man implements Person { private String word; public Man(String word){ this.word = word; } public Man(){ public void say(){ System.out.println("Man Can Say " + word); public class ManJDKProxy implements InvocationHandler { /** * 需要的代理對(duì)象 */ private Object o; public Object bind(Object o){ this.o = o; return Proxy.newProxyInstance(o.getClass().getClassLoader(), o.getClass().getInterfaces(), this); @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("JDK Proxy Design"); return method.invoke(o, args); /** * JDK動(dòng)態(tài)代理 public class ProxyDesign_2 { public static void main(String[] args) { Man man = new Man("Hello"); Person p = (Person)new ManJDKProxy().bind(man); p.say();
* JDK動(dòng)態(tài)代理的優(yōu)缺點(diǎn)
優(yōu)點(diǎn):
1. JDK動(dòng)態(tài)代理是JDK原生的,不需要任何依賴即可使用
2. 通過(guò)反射機(jī)制生成代理類的速度要比CGLib操作字節(jié)碼生成代理類的速度更快
缺點(diǎn):
1. 如果要使用JDK動(dòng)態(tài)代理,被代理的類必須實(shí)現(xiàn)了接口,否則無(wú)法代理(InvocationHandler)
2. JDK動(dòng)態(tài)代理無(wú)法為沒(méi)有在接口中定義的方法實(shí)現(xiàn)代理
3. JDK動(dòng)態(tài)代理執(zhí)行代理方法時(shí),需要通過(guò)反射機(jī)制進(jìn)行回調(diào),此時(shí)方法執(zhí)行的效率比較低
2. CGLIB動(dòng)態(tài)代理
若需要代理的類沒(méi)有實(shí)現(xiàn)接口,JDK的動(dòng)態(tài)代理就無(wú)法使用,Spring會(huì)使用CGLiB動(dòng)態(tài)代理來(lái)生成代理對(duì)象。CGLiB直接操作字節(jié)碼,生成類的子類,重寫類的方法完成代理。
/* * 接口類 */ public interface Person { void say(); } * 接口實(shí)現(xiàn)類 public class Man implements Person { private String word; public Man(String word){ this.word = word; } public Man(){ public void say(){ System.out.println("Man Can Say " + word); public class ManCGLIBProxy { public Object bind(Object target){ Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(target.getClass()); enhancer.setCallback(new MethodInterceptor() { @Override public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { System.out.println("CGLIB Proxy Design"); return method.invoke(target, objects); } }); return enhancer.create(); /** * CGLIB動(dòng)態(tài)代理 public class ProxyDesign_3 { public static void main(String[] args) { Man man = new Man("Hello"); Person p = (Person)new ManCGLIBProxy().bind(man); p.say();
* CGLiB動(dòng)態(tài)代理的優(yōu)缺點(diǎn)
優(yōu)點(diǎn):
1. 使用CGLiB代理的類,不需要實(shí)現(xiàn)接口,因?yàn)镃GLib生成的代理類是直接繼承自需要被代理的類
2. 因?yàn)镃GLiB實(shí)現(xiàn)方式是重寫父類的方法,所以對(duì)final方法,或者private方法是沒(méi)有辦法代理的
3. CGLiB是通過(guò)修改字節(jié)碼生成的代理類,所以CGLib執(zhí)行代理方法的效率要高于JDK的動(dòng)態(tài)代理
缺點(diǎn):
1. 因?yàn)镃GLiB實(shí)現(xiàn)方式是重寫父類的方法,所以對(duì)final方法,或者private方法是沒(méi)有辦法代理的
2. 因?yàn)镃GLiB生成代理類的方式是通過(guò)操作字節(jié)碼(asm工具包),這種生成的代理類的方式比JDK通過(guò)反射生成代理類的方式的效率低
3. Spring項(xiàng)目中如何強(qiáng)制使用CGLIB代理方式
* xml方式
<!-- aop:config用來(lái)在xml中配置切面,指定proxy-target-class="true" --> <aop:config proxy-target-class="true"> <!-- AOP相關(guān)配置 --> </aop:config>
* @Aspect注解方式
<!-- 將proxy-target-class配置設(shè)置為true --> <aop:aspectj-autoproxy proxy-target-class="true"/>
* 配置類注解方式
添加@EnableAspectJAutoProxy(proxyTargetClass = true)
到此這篇關(guān)于SpringAop實(shí)現(xiàn)原理及代理模式的文章就介紹到這了,更多相關(guān)SpringAop代理模式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java中的Semaphore信號(hào)量簡(jiǎn)析
這篇文章主要介紹了Java中的Semaphore信號(hào)量簡(jiǎn)析,Semaphore:信號(hào)量,用來(lái)限制能同時(shí)訪問(wèn)共享資源的線程上限,使用Semaphore實(shí)現(xiàn)簡(jiǎn)單連接池,對(duì)比享元模式下的實(shí)現(xiàn)(用wait和notify),性能和可讀性要更好,需要的朋友可以參考下2023-12-12新版本IntelliJ IDEA 構(gòu)建maven,并用Maven創(chuàng)建一個(gè)web項(xiàng)目(圖文教程)
這篇文章主要介紹了新版本IntelliJ IDEA 構(gòu)建maven,并用Maven創(chuàng)建一個(gè)web項(xiàng)目的圖文教程,需要的朋友可以參考下2018-01-01SpringBoot如何通過(guò)Feign調(diào)用傳遞Header中參數(shù)
這篇文章主要介紹了SpringBoot通過(guò)Feign調(diào)用傳遞Header中參數(shù),本文給大家分享兩種解決方案給大家詳細(xì)講解,需要的朋友可以參考下2023-04-04Feign?請(qǐng)求動(dòng)態(tài)URL方式
這篇文章主要介紹了Feign?請(qǐng)求動(dòng)態(tài)URL方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07使用Spring MVC實(shí)現(xiàn)雙向數(shù)據(jù)綁定
Spring MVC是一個(gè)廣泛用于構(gòu)建Java Web應(yīng)用程序的框架,它提供了眾多功能,包括雙向數(shù)據(jù)綁定,在這篇文章中,我們將向Java新手介紹如何使用Spring MVC實(shí)現(xiàn)雙向數(shù)據(jù)綁定,以及為什么這個(gè)特性如此重要,需要的朋友可以參考下2024-01-01501 Command "HELO" requires an argument問(wèn)題的解決方法
換一個(gè)windows服務(wù)器,發(fā)現(xiàn)就沒(méi)這樣的問(wèn)題,僅在一臺(tái)Linux服務(wù)器上可以重現(xiàn),直觀感覺(jué)就是這臺(tái)Linux服務(wù)器某些配置有問(wèn)題2013-08-08