Java動態代理語法Proxy類原理詳解
1、前言
寫動態代理的程式碼涉及了一個非常重要的類 Proxy,通過Proxy的靜態方法newProxyInstance才會動態建立代理物件。
2、newProxyInstance方法
public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
三個引數分別表示: loader表示類載入器, interfaces表示程式碼要用來代理的介面 , h表示一個 InvocationHandler 物件,前面兩個引數容易理解,
最後一個InvocationHandler是什麼?
InvocationHandler是一個介面,官方文件解釋說,每個代理的例項都有一個與之關聯的 InvocationHandler 實現類,如果代理的方法被呼叫,那麼代理便會通知和轉發給內部的 InvocationHandler 實現類,由它決定處理。
public interface InvocationHandler { public Object invoke(Object proxy,Method method,Object[] args) throws Throwable; }
InvocationHandler 內部只有一個 invoke() 方法,正是這個方法決定了怎麼樣處理代理傳遞過來的方法呼叫。其中引數proxy表示代理物件,method表示代理物件呼叫的方法,args表示呼叫的方法中的引數。所以Proxy動態產生的代理會呼叫InvocationHandler實現類,所以InvocationHandler才是實際執行者。
3、程式碼例項
//抽象主題 interface AbstractSubject { void request(); } //真實主題 class RealSubject implements AbstractSubject { public void request() { System.out.println("訪問真實主題方法..."); } } //真實主題 class RealSubject1 implements AbstractSubject { public void request() { System.out.println("訪問真實主題方法1..."); } } //動態代理類 class DynamicProxy implements InvocationHandler { private Object object; public DynamicProxy(Object object) { this.object = object; } @Override public Object invoke(Object proxy,Object[] args) throws Throwable { before(); Object result = method.invoke(object,args); after(); return result; } private void before() { System.out.println("hello!"); } private void after() { System.out.println("bye!"); } } //測試類 public class TestDynamicProxyPattern { public static void main(String[] args) { AbstractSubject abstractSubject = new RealSubject(); DynamicProxy dynamicProxy = new DynamicProxy(abstractSubject); Proxy.newProxyInstance(abstractSubject.getClass().getClassLoader(),abstractSubject.getClass().getInterfaces(),dynamicProxy); abstractProxy.request(); } }
上述動態代理只需要傳入需要被代理類的物件(DynamicProxy dynamicProxy = new DynamicProxy(abstractSubject)),然後呼叫Proxy類的工廠方法newProxyInstance去動態地建立一個代理類,最後呼叫代理類的方法便實現了“增強功能”。使用了動態代理之後,無論有多少類多少方法需要增加邏輯,只需要在使用的時候將類物件傳入得到代理物件,然後使用代理物件呼叫需要增強的方法即可。
所以這時候如果增加一個實現抽象主題的真是主題類,比如說叫做RealSubject1,這個時候只要只要把該類的物件傳入動態代理類DynamicProxy中,通過介面又可以實現介面AbstractSubject的實現類。
這樣來實現:
AbstractSubject abstractSubject1 = new RealSubject1();
DynamicProxy dynamicProxy = new DynamicProxy(abstractSubject1);
4、總結
1、區別於靜態代理的生成代理類,動態代理的代理類通過 Proxy.newInstance() 方法生成。靜態代理和動態代理的區別是在於要不要開發者自己定義 Proxy 類。
2、不管是靜態代理還是動態代理,代理與被代理者都要實現介面,還是要是面向介面程式設計,目的都是增強現有功能。
3、動態代理通過 Proxy 動態生成 proxy class,但是它也指定了一個 InvocationHandler 的實現類。
4、動態代理也有缺陷,它要求需要代理的物件必須實現了某個介面,而且也不夠靈活,動態代理會為介面中的宣告的所有方法新增上相同的代理邏輯。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。