SpringAop實現(xiàn)原理及代理模式詳解
Spring Aop的原理
Spring的AOP就是通過動態(tài)代理實現(xiàn)的。當(dāng)為某個Bean或者某些Bean配置切面時,Spring會為其創(chuàng)建代理對象,當(dāng)調(diào)用該對象的某個方法時,實際是調(diào)用生成的代理類的對象方法。Spring的Aop主要是使用了兩個動態(tài)代理,分別是JDK的動態(tài)代理和CGLIB動態(tài)代理。
1. JDK動態(tài)代理
如果代理類實現(xiàn)了接口,Spring默認(rèn)會使用JDK動態(tài)代理。JDK的動態(tài)代理是基于反射實現(xiàn)。JDK通過反射,生成一個代理類,這個代理類實現(xiàn)了原來那個類的全部接口,并對接口中定義的所有方法進(jìn)行了代理。當(dāng)我們通過代理對象執(zhí)行原來那個類的方法時,代理類底層會通過反射機(jī)制,調(diào)用我們實現(xiàn)的InvocationHandler接口的invoke方法。
/* * 接口類 */ public interface Person { void say(); } * 接口實現(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 { /** * 需要的代理對象 */ 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動態(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動態(tài)代理的優(yōu)缺點
優(yōu)點:
1. JDK動態(tài)代理是JDK原生的,不需要任何依賴即可使用
2. 通過反射機(jī)制生成代理類的速度要比CGLib操作字節(jié)碼生成代理類的速度更快
缺點:
1. 如果要使用JDK動態(tài)代理,被代理的類必須實現(xiàn)了接口,否則無法代理(InvocationHandler)
2. JDK動態(tài)代理無法為沒有在接口中定義的方法實現(xiàn)代理
3. JDK動態(tài)代理執(zhí)行代理方法時,需要通過反射機(jī)制進(jìn)行回調(diào),此時方法執(zhí)行的效率比較低
2. CGLIB動態(tài)代理
若需要代理的類沒有實現(xiàn)接口,JDK的動態(tài)代理就無法使用,Spring會使用CGLiB動態(tài)代理來生成代理對象。CGLiB直接操作字節(jié)碼,生成類的子類,重寫類的方法完成代理。
/* * 接口類 */ public interface Person { void say(); } * 接口實現(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動態(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動態(tài)代理的優(yōu)缺點
優(yōu)點:
1. 使用CGLiB代理的類,不需要實現(xiàn)接口,因為CGLib生成的代理類是直接繼承自需要被代理的類
2. 因為CGLiB實現(xiàn)方式是重寫父類的方法,所以對final方法,或者private方法是沒有辦法代理的
3. CGLiB是通過修改字節(jié)碼生成的代理類,所以CGLib執(zhí)行代理方法的效率要高于JDK的動態(tài)代理
缺點:
1. 因為CGLiB實現(xiàn)方式是重寫父類的方法,所以對final方法,或者private方法是沒有辦法代理的
2. 因為CGLiB生成代理類的方式是通過操作字節(jié)碼(asm工具包),這種生成的代理類的方式比JDK通過反射生成代理類的方式的效率低
3. Spring項目中如何強(qiáng)制使用CGLIB代理方式
* xml方式
<!-- aop:config用來在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實現(xiàn)原理及代理模式的文章就介紹到這了,更多相關(guān)SpringAop代理模式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
新版本IntelliJ IDEA 構(gòu)建maven,并用Maven創(chuàng)建一個web項目(圖文教程)
這篇文章主要介紹了新版本IntelliJ IDEA 構(gòu)建maven,并用Maven創(chuàng)建一個web項目的圖文教程,需要的朋友可以參考下2018-01-01SpringBoot如何通過Feign調(diào)用傳遞Header中參數(shù)
這篇文章主要介紹了SpringBoot通過Feign調(diào)用傳遞Header中參數(shù),本文給大家分享兩種解決方案給大家詳細(xì)講解,需要的朋友可以參考下2023-04-04使用Spring MVC實現(xiàn)雙向數(shù)據(jù)綁定
Spring MVC是一個廣泛用于構(gòu)建Java Web應(yīng)用程序的框架,它提供了眾多功能,包括雙向數(shù)據(jù)綁定,在這篇文章中,我們將向Java新手介紹如何使用Spring MVC實現(xiàn)雙向數(shù)據(jù)綁定,以及為什么這個特性如此重要,需要的朋友可以參考下2024-01-01501 Command "HELO" requires an argument問題的解決方法
換一個windows服務(wù)器,發(fā)現(xiàn)就沒這樣的問題,僅在一臺Linux服務(wù)器上可以重現(xiàn),直觀感覺就是這臺Linux服務(wù)器某些配置有問題2013-08-08