詳解Java JDK動(dòng)態(tài)代理
今天來(lái)看看Java的另一種代理方式——JDK動(dòng)態(tài)代理
我們之前所介紹的代理方式叫靜態(tài)代理,也就是靜態(tài)的生成代理對(duì)象,而動(dòng)態(tài)代理則是在運(yùn)行時(shí)創(chuàng)建代理對(duì)象。動(dòng)態(tài)代理有更強(qiáng)大的攔截請(qǐng)求功能,因?yàn)榭梢垣@得類(lèi)的運(yùn)行時(shí)信息,可以根據(jù)運(yùn)行時(shí)信息來(lái)獲得更為強(qiáng)大的執(zhí)(騷)行(操)力(作)。
我們還是以上一個(gè)例子為例,這里的IStars接口和Stars類(lèi)都不需要修改,只需要修改代理類(lèi)。
創(chuàng)建JDK動(dòng)態(tài)代理需要先實(shí)現(xiàn)InvocationHandler接口,并重寫(xiě)其中的invoke方法,具體步驟如下:
1. 創(chuàng)建一個(gè)類(lèi)實(shí)現(xiàn)InvocationHandler接口。
2. 給Proxy類(lèi)提供委托類(lèi)的ClassLoader和Interfaces來(lái)創(chuàng)建動(dòng)態(tài)代理類(lèi)。
3. 利用反射機(jī)制得到動(dòng)態(tài)代理類(lèi)的構(gòu)造函數(shù)。
4. 利用動(dòng)態(tài)代理類(lèi)的構(gòu)造函數(shù)創(chuàng)建動(dòng)態(tài)代理類(lèi)對(duì)象。
我們用動(dòng)態(tài)代理來(lái)改造一下之前的類(lèi):
接口和委托類(lèi)不需要修改:
public interface IStars { void sing(); void dance(); }
public class Stars implements IStars{ private String name; public Stars(String name) { this.name = name; } public String getName() { return name; } public void setName(String name) { this.name = name; } public void sing(){ System.out.println(getName() + " 唱了一首歌."); } public void dance(){ System.out.println(getName() + " 跳了一支舞."); } }
這是使用動(dòng)態(tài)代理后的代理類(lèi):
public class StarsNewProxy implements InvocationHandler { //代理類(lèi)持有委托類(lèi)的對(duì)象引用 private Object object; //保存sing和dance的次數(shù) private int num; public StarsNewProxy(Object object){ this.object = object; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { if (!runBefore(method)){ return null; }; //利用反射機(jī)制將請(qǐng)求分派給委托類(lèi)處理,Method的invoke返回Object對(duì)象作為方法執(zhí)行結(jié)果 Object result = method.invoke(object,args); runAfter(method); return result; } private boolean runBefore(Method method){ System.out.println("我是代理,攔截到請(qǐng)求"); if (method.getName().equals("dance")){ System.out.println("抱歉,明星腳受傷了,不能跳舞表演了。"); return false; } return true; } private void runAfter(Method method){ System.out.println("我是代理,請(qǐng)求處理完畢"); } }
新建一個(gè)工廠類(lèi)來(lái)返回代理實(shí)例:
public class StarsNewProxyFactory { //構(gòu)建工廠類(lèi),客戶類(lèi)調(diào)用此方法獲得代理對(duì)象 //對(duì)于客戶類(lèi)而言,代理類(lèi)對(duì)象和委托類(lèi)對(duì)象是一樣的,不需要知道具體返回的類(lèi)型 public static IStars getInstance(String name){ IStars stars = new Stars(name); InvocationHandler handler = new StarsNewProxy(stars); IStars proxy = null; proxy = (IStars) Proxy.newProxyInstance( stars.getClass().getClassLoader(), stars.getClass().getInterfaces(), handler ); return proxy; } }
改寫(xiě)一下測(cè)試類(lèi):
public class Test { public static void main(String[] args){ // testA(); testB(); } /** * 靜態(tài)代理 */ private static void testA(){ //創(chuàng)建目標(biāo)對(duì)象 IStars stars = new Stars("Frank"); //代理對(duì)象,把目標(biāo)傳給代理對(duì)象,建立關(guān)系 IStars starsProxy = new StarsProxy(stars); for (int i = 0;i < 5; i++){ starsProxy.sing(); } } /** * JDK動(dòng)態(tài)代理 */ private static void testB(){ IStars proxy = StarsNewProxyFactory.getInstance("Frank"); proxy.dance(); proxy.sing(); } }
輸出如下:
我是代理,攔截到請(qǐng)求
抱歉,明星腳受傷了,不能跳舞表演了。
我是代理,攔截到請(qǐng)求
Frank 唱了一首歌.
我是代理,請(qǐng)求處理完畢
使用動(dòng)態(tài)代理時(shí)實(shí)現(xiàn)了InvocationHandler接口并重寫(xiě)了invoke方法,invoke方法的三個(gè)參數(shù):
Object invoke(Object proxy, Method method, Object[] args) throws Throwable proxy: 被代理的對(duì)象 method: 被代理對(duì)象的某個(gè)方法的Method對(duì)象 args: 被代理對(duì)象的某個(gè)方法接受的參數(shù)
Proxy的newProxyInstance方法詳情如下:
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException loader: 一個(gè)ClassLoader對(duì)象,定義了由哪個(gè)ClassLoader對(duì)象來(lái)對(duì)生成的代理對(duì)象進(jìn)行加載 interfaces: 一個(gè)Interface對(duì)象的數(shù)組,表示的是我將要給我需要代理的對(duì)象提供一組什么接口,如果我提供了一組接口給它,那么這個(gè)代理對(duì)象就宣稱實(shí)現(xiàn)了該接口(多態(tài)),這樣我就能調(diào)用這組接口中的方法了 h: 一個(gè)InvocationHandler對(duì)象,表示的是當(dāng)我這個(gè)動(dòng)態(tài)代理對(duì)象在調(diào)用方法的時(shí)候,會(huì)關(guān)聯(lián)到哪一個(gè)InvocationHandler對(duì)象上
可以看到,這里的動(dòng)態(tài)代理跟靜態(tài)代理一樣,在代理類(lèi)內(nèi)部保存了一個(gè)委托類(lèi)的實(shí)例,實(shí)際上都是調(diào)用原來(lái)的委托實(shí)例來(lái)進(jìn)行需要的操作,代理類(lèi)相當(dāng)于給委托類(lèi)加上一個(gè)外殼,把委托類(lèi)置于代理類(lèi)的內(nèi)部,從而可以控制客戶類(lèi)對(duì)委托類(lèi)的訪問(wèn),就像上例中,代理類(lèi)攔截了客戶類(lèi)對(duì)Stars類(lèi)的dance方法的訪問(wèn),并且輸出了補(bǔ)充信息。
動(dòng)態(tài)代理跟靜態(tài)代理最大的不同便是生成代理類(lèi)的時(shí)期不同,靜態(tài)代理是在編譯期,而動(dòng)態(tài)代理則是在運(yùn)行時(shí)根據(jù)委托類(lèi)信息動(dòng)態(tài)生成。
其次,動(dòng)態(tài)代理實(shí)現(xiàn)的是InvocationHandler接口,而靜態(tài)代理則是直接實(shí)現(xiàn)公共接口。當(dāng)然動(dòng)態(tài)代理也是需要實(shí)現(xiàn)相同的接口的,只是將接口信息放在了getInstance內(nèi)部,相當(dāng)于代理類(lèi)跟委托類(lèi)之間的約定,“這幾個(gè)方法幫我代理一下吧”。
最后,動(dòng)態(tài)代理可以獲得更多的運(yùn)行時(shí)信息,使用起來(lái)也會(huì)更加靈活。
至此,JDK動(dòng)態(tài)代理講解完畢,歡迎大家繼續(xù)關(guān)注!
以上就是詳解Java JDK動(dòng)態(tài)代理的詳細(xì)內(nèi)容,更多關(guān)于Java JDK動(dòng)態(tài)代理的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- 深入理解java動(dòng)態(tài)代理的兩種實(shí)現(xiàn)方式(JDK/Cglib)
- java動(dòng)態(tài)代理(jdk與cglib)詳細(xì)解析
- Java JDK動(dòng)態(tài)代理(AOP)的實(shí)現(xiàn)原理與使用詳析
- java jdk動(dòng)態(tài)代理詳解
- Java JDK動(dòng)態(tài)代理的基本原理詳細(xì)介紹
- Java JDK 動(dòng)態(tài)代理的使用方法示例
- java代理 jdk動(dòng)態(tài)代理應(yīng)用案列
- Java JDK動(dòng)態(tài)代理實(shí)現(xiàn)原理實(shí)例解析
- Java中JDK動(dòng)態(tài)代理的超詳細(xì)講解
相關(guān)文章
Springboot項(xiàng)目實(shí)現(xiàn)Mysql多數(shù)據(jù)源切換的完整實(shí)例
這篇文章主要給大家介紹了關(guān)于Springboot項(xiàng)目實(shí)現(xiàn)Mysql多數(shù)據(jù)源切換的完整實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11java實(shí)現(xiàn)往hive 的map類(lèi)型字段寫(xiě)數(shù)據(jù)
這篇文章主要介紹了java實(shí)現(xiàn)往hive 的map類(lèi)型字段寫(xiě)數(shù)據(jù)操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07SpringBoot 配置文件加載位置與優(yōu)先級(jí)問(wèn)題詳解
這篇文章主要介紹了SpringBoot 配置文件加載位置與優(yōu)先級(jí)問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-09-09