1. 程式人生 > 程式設計 >Java動態代理語法Proxy類原理詳解

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、動態代理也有缺陷,它要求需要代理的物件必須實現了某個介面,而且也不夠靈活,動態代理會為介面中的宣告的所有方法新增上相同的代理邏輯。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。