java入門(mén)學(xué)習(xí)提升第十一篇: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è)工廠(chǎng)類(lèi)來(lái)返回代理實(shí)例:

public class StarsNewProxyFactory {

//構(gòu)建工廠(chǎng)類(lèi),客戶(hù)類(lèi)調(diào)用此方法獲得代理對(duì)象

//對(duì)于客戶(hù)類(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ì)象就宣稱(chēng)實(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)部,從而可以控制客戶(hù)類(lèi)對(duì)委托類(lèi)的訪(fǎng)問(wèn),就像上例中,代理類(lèi)攔截了客戶(hù)類(lèi)對(duì)Stars類(lèi)的dance方法的訪(fǎng)問(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)注!

我做開(kāi)發(fā)十多年的時(shí)間,如果大家對(duì)于學(xué)習(xí)java的學(xué)習(xí)方法,學(xué)習(xí)路線(xiàn)以及你不知道自己應(yīng)該是自學(xué)還是培訓(xùn)的疑問(wèn),都可以隨時(shí)來(lái)問(wèn)我,大家可以加我的java交流學(xué)習(xí)qun:四九四,八零一,九三一,qun內(nèi)有學(xué)習(xí)教程以及開(kāi)發(fā)工具

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀(guān)點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容