JAVA中的靜態(tài)代理、動(dòng)態(tài)代理以及CGLIB動(dòng)態(tài)代理總結(jié)
代理模式是java中最常用的設(shè)計(jì)模式之一,尤其是在spring框架中廣泛應(yīng)用。對(duì)于java的代理模式,一般可分為:靜態(tài)代理、動(dòng)態(tài)代理、以及CGLIB實(shí)現(xiàn)動(dòng)態(tài)代理。
對(duì)于上述三種代理模式,分別進(jìn)行說(shuō)明。
1.靜態(tài)代理
靜態(tài)代理其實(shí)就是在程序運(yùn)行之前,提前寫好被代理方法的代理類,編譯后運(yùn)行。在程序運(yùn)行之前,class已經(jīng)存在。
下面我們實(shí)現(xiàn)一個(gè)靜態(tài)代理demo:
靜態(tài)代理
定義一個(gè)接口Target
package com.test.proxy; public interface Target { public String execute(); }
TargetImpl 實(shí)現(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; } }
測(cè)試類:
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); } }
運(yùn)行結(jié)果:
perProcess TargetImpl execute! postProcess execute
靜態(tài)代理需要針對(duì)被代理的方法提前寫好代理類,如果被代理的方法非常多則需要編寫很多代碼,因此,對(duì)于上述缺點(diǎn),通過(guò)動(dòng)態(tài)代理的方式進(jìn)行了彌補(bǔ)。
2.動(dòng)態(tài)代理
動(dòng)態(tài)代理主要是通過(guò)反射機(jī)制,在運(yùn)行時(shí)動(dòng)態(tài)生成所需代理的class.
動(dòng)態(tài)代理
接口
package com.test.dynamic; public interface Target { public String execute(); }
實(shí)現(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; } }
測(cè)試類
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); } }
運(yùn)行結(jié)果:
========before========== TargetImpl execute! ========after=========== execute
無(wú)論是動(dòng)態(tài)代理還是靜態(tài)帶領(lǐng),都需要定義接口,然后才能實(shí)現(xiàn)代理功能。這同樣存在局限性,因此,為了解決這個(gè)問(wèn)題,出現(xiàn)了第三種代理方式:cglib代理。
3.cglib代理
CGLib采用了非常底層的字節(jié)碼技術(shù),其原理是通過(guò)字節(jié)碼技術(shù)為一個(gè)類創(chuàng)建子類,并在子類中采用方法攔截的技術(shù)攔截所有父類方法的調(diào)用,順勢(shì)織入橫切邏輯。JDK動(dòng)態(tài)代理與CGLib動(dòng)態(tài)代理均是實(shí)現(xiàn)Spring AOP的基礎(chǔ)。
cglib動(dòng)態(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"; } }
測(cè)試類
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
代理對(duì)象的生成過(guò)程由Enhancer類實(shí)現(xiàn),大概步驟如下:
1、生成代理類Class的二進(jìn)制字節(jié)碼;
2、通過(guò)Class.forName加載二進(jìn)制字節(jié)碼,生成Class對(duì)象;
3、通過(guò)反射機(jī)制獲取實(shí)例構(gòu)造,并初始化代理類對(duì)象。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Java靜態(tài)代理和動(dòng)態(tài)代理的深入講解
- Java動(dòng)態(tài)代理靜態(tài)代理實(shí)例分析
- Java代理模式實(shí)例詳解【靜態(tài)代理與動(dòng)態(tài)代理】
- Java靜態(tài)代理和動(dòng)態(tài)代理總結(jié)
- 淺談Java代理(jdk靜態(tài)代理、動(dòng)態(tài)代理和cglib動(dòng)態(tài)代理)
- 深入解析java中的靜態(tài)代理與動(dòng)態(tài)代理
- java 靜態(tài)代理 動(dòng)態(tài)代理深入學(xué)習(xí)
- 淺談Java 代理機(jī)制
相關(guān)文章
圖文詳解java反射機(jī)制及常用應(yīng)用場(chǎng)景
這篇文章主要為大家介紹了圖文詳解java反射機(jī)制及常用應(yīng)用場(chǎng)景,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-03-03詳解Java如何優(yōu)雅的調(diào)用dubbo同時(shí)不使用其它jar包
這篇文章主要介紹了如何在不使用他人jar包的情況下優(yōu)雅的進(jìn)行dubbo調(diào)用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧2023-02-02基于mybatis-plus QueryWrapper 排序的坑
這篇文章主要介紹了mybatis-plus QueryWrapper 排序的坑,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01使用java swing實(shí)現(xiàn)qq登錄界面示例分享
這篇文章主要介紹了使用java swing實(shí)現(xiàn)qq登錄界面示例,需要的朋友可以參考下2014-04-04java事件處理模型知識(shí)點(diǎn)總結(jié)
在本篇文章里小辮給大家分享的是一篇關(guān)于java事件處理模型知識(shí)點(diǎn)總結(jié)內(nèi)容,有興趣的朋友們可以學(xué)習(xí)下。2021-01-01Linux(centos7)安裝jdk1.8的詳細(xì)步驟
Linux的使用相信大家都要用到j(luò)ava吧,在使用java前我們得先安裝jdk以及配置環(huán)境變量等工作,下面這篇文章主要給大家介紹了關(guān)于Linux(centos7)安裝jdk1.8的詳細(xì)步驟,需要的朋友可以參考下2023-10-10Redis緩存,泛型集合與json字符串的相互轉(zhuǎn)換實(shí)例
這篇文章主要介紹了Redis緩存,泛型集合與json字符串的相互轉(zhuǎn)換實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-07-07