這期內(nèi)容當(dāng)中小編將會給大家?guī)碛嘘P(guān)JAVA中靜態(tài)代理與動態(tài)代理的區(qū)別有哪些,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。
網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)公司!專注于網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、微信小程序開發(fā)、集團企業(yè)網(wǎng)站建設(shè)等服務(wù)項目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了橫峰免費建站歡迎大家使用!
代理模式是java中最常用的設(shè)計模式之一,尤其是在spring框架中廣泛應(yīng)用。對于java的代理模式,一般可分為:靜態(tài)代理、動態(tài)代理、以及CGLIB實現(xiàn)動態(tài)代理。
對于上述三種代理模式,分別進行說明。
1.靜態(tài)代理
靜態(tài)代理其實就是在程序運行之前,提前寫好被代理方法的代理類,編譯后運行。在程序運行之前,class已經(jīng)存在。
下面我們實現(xiàn)一個靜態(tài)代理demo:
靜態(tài)代理
定義一個接口Target
package com.test.proxy; public interface Target { public String execute(); }
TargetImpl 實現(xiàn)接口Target
package com.test.proxy; public class TargetImpl implements Target { @Override public String execute() { System.out.println("TargetImpl execute!"); return "execute"; } }
代理類
package com.test.proxy; public class Proxy implements Target{ private Target target; public Proxy(Target target) { this.target = target; } @Override public String execute() { System.out.println("perProcess"); String result = this.target.execute(); System.out.println("postProcess"); return result; } }
測試類:
package com.test.proxy; public class ProxyTest { public static void main(String[] args) { Target target = new TargetImpl(); Proxy p = new Proxy(target); String result = p.execute(); System.out.println(result); } }
運行結(jié)果:
perProcess TargetImpl execute! postProcess execute
靜態(tài)代理需要針對被代理的方法提前寫好代理類,如果被代理的方法非常多則需要編寫很多代碼,因此,對于上述缺點,通過動態(tài)代理的方式進行了彌補。
2.動態(tài)代理
動態(tài)代理主要是通過反射機制,在運行時動態(tài)生成所需代理的class.
動態(tài)代理
接口
package com.test.dynamic; public interface Target { public String execute(); }
實現(xiàn)類
package com.test.dynamic; public class TargetImpl implements Target { @Override public String execute() { System.out.println("TargetImpl execute!"); return "execute"; } }
代理類
package com.test.dynamic; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; public class DynamicProxyHandler implements InvocationHandler{ private Target target; public DynamicProxyHandler(Target target) { this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("========before=========="); Object result = method.invoke(target,args); System.out.println("========after==========="); return result; } }
測試類
package com.test.dynamic; import java.lang.reflect.Proxy; public class DynamicProxyTest { public static void main(String[] args) { Target target = new TargetImpl(); DynamicProxyHandler handler = new DynamicProxyHandler(target); Target proxySubject = (Target) Proxy.newProxyInstance(TargetImpl.class.getClassLoader(),TargetImpl.class.getInterfaces(),handler); String result = proxySubject.execute(); System.out.println(result); } }
運行結(jié)果:
========before========== TargetImpl execute! ========after=========== execute
無論是動態(tài)代理還是靜態(tài)帶領(lǐng),都需要定義接口,然后才能實現(xiàn)代理功能。這同樣存在局限性,因此,為了解決這個問題,出現(xiàn)了第三種代理方式:cglib代理。
3.cglib代理
CGLib采用了非常底層的字節(jié)碼技術(shù),其原理是通過字節(jié)碼技術(shù)為一個類創(chuàng)建子類,并在子類中采用方法攔截的技術(shù)攔截所有父類方法的調(diào)用,順勢織入橫切邏輯。JDK動態(tài)代理與CGLib動態(tài)代理均是實現(xiàn)Spring AOP的基礎(chǔ)。
cglib動態(tài)代理
目標(biāo)類
package com.test.cglib; public class Target { public String execute() { String message = "-----------test------------"; System.out.println(message); return message; } }
通用代理類:
package com.test.cglib; import net.sf.cglib.proxy.MethodInterceptor; import net.sf.cglib.proxy.MethodProxy; import java.lang.reflect.Method; public class MyMethodInterceptor implements MethodInterceptor{ @Override public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { System.out.println(">>>>MethodInterceptor start..."); Object result = proxy.invokeSuper(obj,args); System.out.println(">>>>MethodInterceptor ending..."); return "result"; } }
測試類
package com.test.cglib; import net.sf.cglib.proxy.Enhancer; public class CglibTest { public static void main(String ... args) { System.out.println("***************"); Target target = new Target(); CglibTest test = new CglibTest(); Target proxyTarget = (Target) test.createProxy(Target.class); String res = proxyTarget.execute(); System.out.println(res); } public Object createProxy(Class targetClass) { Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(targetClass); enhancer.setCallback(new MyMethodInterceptor()); return enhancer.create(); } }
執(zhí)行結(jié)果:
*************** >>>>MethodInterceptor start... -----------test------------ >>>>MethodInterceptor ending... result
代理對象的生成過程由Enhancer類實現(xiàn),大概步驟如下:
1、生成代理類Class的二進制字節(jié)碼;
2、通過Class.forName加載二進制字節(jié)碼,生成Class對象;
3、通過反射機制獲取實例構(gòu)造,并初始化代理類對象。
上述就是小編為大家分享的JAVA中靜態(tài)代理與動態(tài)代理的區(qū)別有哪些了,如果剛好有類似的疑惑,不妨參照上述分析進行理解。如果想知道更多相關(guān)知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。