更新時間:2019-08-11 09:00:00 來源:動力節點 瀏覽2120次
在處理數據中,在數據量少的情況下,靜態代理完全可以應付,但是當數據量負責程度已經無法使用靜態代理完成的時候,動態代理就顯得非常必要了。
1、什么是動態代理
代理類在程序運行時創建的代理方式被成為動態代理。也就是說,這種情況下,代理類并不是在Java代碼中定義的,而是在運行時根據我們在Java代碼中的“指示”動態生成的。相比于靜態代理,動態代理的優勢在于可以很方便的對代理類的函數進行統一的處理,而不用修改每個代理類的函數。
2、使用動態代理
(1)InvocationHandler接口
在使用動態代理時,我們需要定義一個位于代理類與委托類之間的中介類,這個中介類被要求實現InvocationHandler接口,這個接口的定義如下:
public interface InvocationHandler {
Object invoke(Object proxy, Method method, Object[] args);
}
從InvocationHandler這個名稱我們就可以知道,實現了這個接口的中介類用做“調用處理器”。當我們調用代理類對象的方法時,這個“調用”會轉送到invoke方法中,代理類對象作為proxy參數傳入,參數method標識了我們具體調用的是代理類的哪個方法,args為這個方法的參數。這樣一來,我們對代理類中的所有方法的調用都會變為對invoke的調用,這樣我們可以在invoke方法中添加統一的處理邏輯(也可以根據method參數對不同的代理類方法做不同的處理)。因此我們只需在中介類的invoke方法實現中輸出“before”,然后調用委托類的invoke方法,再輸出“after”。下面我們來一步一步具體實現它。
(2)委托類的定義
動態代理方式下,要求委托類必須實現某個接口,這里我們實現的是Sell接口。委托類Vendor類的定義如下:
public class Vendor implements Sell {
public void sell() {
System.out.println("In sell method");
}
public void ad() {
System,out.println("ad method")
}
}
(3)中介類
上面我們提到過,中介類必須實現InvocationHandler接口,作為調用處理器”攔截“對代理類方法的調用。中介類的定義如下:
public class DynamicProxy implements InvocationHandler {
private Object obj; //obj為委托類對象;
public DynamicProxy(Object obj) {
this.obj = obj;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("before");
Object result = method.invoke(obj, args);
System.out.println("after");
return result;
}
}
從以上代碼中我們可以看到,中介類持有一個委托類對象引用,在invoke方法中調用了委托類對象的相應方法(第11行),看到這里是不是覺得似曾相識?通過聚合方式持有委托類對象引用,把外部對invoke的調用最終都轉為對委托類對象的調用。這不就是我們上面介紹的靜態代理的一種實現方式嗎?實際上,中介類與委托類構成了靜態代理關系,在這個關系中,中介類是代理類,委托類就是委托類;
代理類與中介類也構成一個靜態代理關系,在這個關系中,中介類是委托類,代理類是代理類。也就是說,動態代理關系由兩組靜態代理關系組成,這就是動態代理的原理。下面我們來介紹一下如何”指示“以動態生成代理類。
(4)動態生成代理類
動態生成代理類的相關代碼如下:
public class Main {
public static void main(String[] args) {
//創建中介類實例
DynamicProxy inter = new DynamicProxy(new Vendor());
//加上這句將會產生一個$Proxy0.class文件,這個文件即為動態生成的代理類文件
System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles","true");
//獲取代理類實例sell
Sell sell = (Sell)(Proxy.newProxyInstance(Sell.class.getClassLoader(),
new Class[] {Sell.class}, inter));
//通過代理類對象調用代理類方法,實際上會轉到invoke方法調用
sell.sell();
sell.ad();
}
}
在以上代碼中,我們調用Proxy類的newProxyInstance方法來獲取一個代理類實例。這個代理類實現了我們指定的接口并且會把方法調用分發到指定的調用處理器。這個方法的聲明如下:
public static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h) throws IllegalArgumentException
方法的三個參數含義分別如下:
loader:定義了代理類的ClassLoder;
interfaces:代理類實現的接口列表
h:調用處理器,也就是我們上面定義的實現了InvocationHandler接口的類實例
我們運行一下,看看我們的動態代理是否能正常工作。我這里運行后的輸出,說明我們的動態代理確實奏效了。
通過簡單的分析,不知道大家是否已經能夠理解動態代理這個方法的使用。如果想要學習更多的java知識,可以關注動力節點java學院官網,更新精彩內容等你來看。
相關閱讀
0基礎 0學費 15天面授
有基礎 直達就業
業余時間 高薪轉行
工作1~3年,加薪神器
工作3~5年,晉升架構
提交申請后,顧問老師會電話與您溝通安排學習