程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> JAVA綜合教程 >> java動態代理的2種實現方式,java動態代理2種

java動態代理的2種實現方式,java動態代理2種

編輯:JAVA綜合教程

java動態代理的2種實現方式,java動態代理2種


java的動態代理在接java的api上有說明,這裡就不寫了。我理解的代理:

  對特定接口中特定方法的功能進行擴展,這就是代理。代理是通過代理實例關聯的調用處理程序對象調用方法。

下面通過一個例子看一下:

  接口:

public interface Num {
    void show();
    
    int getNum();
    
    int getProduct(int x);
}

  實現類:

public class MyNum implements Num {
    @Override
    public int getNum() {
        return 3;
    }
    
    @Override
    public int getProduct(int x) {
        return x;
    }

    @Override
    public void show() {
        System.out.println("底層方法打印數字99");
    }
}

先看一下Method中的invoke方法在api中是怎麼描述的

  

  就是說調用處理程序對接口的實現類對象調用Method對象表示的底層方法。

  第一種實現代理的方式:

public class NumProxy {
    private Object num;

  //通過構造方法構造接口的實現類對象 public NumProxy(Object num) { this.num = num; } public Object getNumByProxy(){ Object numProxy = Proxy.newProxyInstance(num.getClass().getClassLoader(), new Class[]{Num.class}, new InvocationHandler() { /** * method: 對應於在代理實例上調用的接口方法的 Method 實例。我理解的就是被代理的真實方法實例 * args: 我理解的是真實方法的參數數組 */ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { Object obj = null; System.out.println("在方法之前開始記錄"); String methodName = method.getName(); if("getProduct".equals(methodName)){ obj = method.invoke(num, args); obj = (Integer) obj * 2; System.out.println("proxy: getProduct()結束"); } else if("show".equals(methodName)){ obj = method.invoke(num, args); System.out.println("proxy: show()結束"); } return obj; } }); return numProxy; } }

   第二種實現代理的方式:通過實現InvocationHandler接口

public class NumProxyImpl implements InvocationHandler {
  //這裡我把接口類型具體化了, 沒有寫成Object private Num num; public NumProxyImpl(Num num){ this.num = num; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { Object obj = null; String methodName = method.getName(); if("getProduct".equals(methodName)){ System.out.println("proxy: getProduct()開始"); obj = method.invoke(num, args); obj = (Integer) obj * 2; System.out.println("proxy: getProduct()結束"); }else if("show".equals(methodName)){ System.out.println("proxy: show()開始"); obj = method.invoke(num, args); System.out.println("proxy: show()結束"); } return obj; } }

  測試代碼:

public class TestNum {
    public static void main(String[] args) {
    //兩種方式一起測試 NumProxy np = new NumProxy(new MyNum()); Num numProxy = (Num) np.getNumByProxy(); int x = numProxy.getProduct(2); System.out.println(x); numProxy.show(); System.out.println("----------------"); NumProxyImpl npi = new NumProxyImpl(new MyNum()); Num numPro = (Num) Proxy.newProxyInstance(Num.class.getClassLoader(), new Class[]{Num.class}, npi); int n = numPro.getProduct(3); System.out.println(n); numPro.show(); } }

 控制台結果:

  

第二種方式有點小疑惑,不知道大家有沒有,那就是並沒有顯示的調用NumProxyImpl中的invoke方法,可是卻執行了,嗯嗯,這個自己下去看一下啊

不想麻煩的只需要記住就行了。

 

比如編碼的處理就可以用到代理,下次寫個例子。

 

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved