java動態代理原理(Proxy,InvocationHandler),含$Proxy0源碼

來源:互聯網
上載者:User

 

原文:http://www.2cto.com/kf/201109/103285.html

 今天沒事,把以前的知識回顧一下,網上找了找,發現有一篇比較適合。
概述:其實JDK的動態代理,實際上就是“反射”與“執行時動態產生位元組碼”二者的結合體;就spring的AOP而言也是用的JDK的動態代理(當然也有cglib方式)。
以下來自網路上的部落格:
 
一.相關類及其方法:
java.lang.reflect.Proxy,
Proxy 提供用於建立動態代理類和執行個體的靜態方法.
newProxyInstance()
返回一個指定介面的代理類執行個體,該介面可以將方法調用指派到指定的調用處理常式
(詳見api文檔)

java.lang.reflect.InvocationHandler,
InvocationHandler 是代理執行個體的調用處理常式 實現的介面。
invoke()
在代理執行個體上處理方法調用並返回結果。在與方法關聯的代理執行個體上調用方法時,將在調用處理常式上調用此方法。
(詳見api文檔)
二.原始碼:
被代理對象的介面及實作類別:
 
package com.ml.test;
 
public interface Manager {
public void modify();
}
 
package com.ml.test;
 
public class ManagerImpl implements Manager {
 
@Override
public void modify() {
   System.out.println("*******modify()方法被調用");
}
}
業務代理類:
package com.ml.test;
 
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
 
public class BusinessHandler implements InvocationHandler {
 
private Object object = null;
 
public BusinessHandler(Object object) {
   this.object = object;
}
 
@Override
public Object invoke(Object proxy, Method method, Object[] args)
    throws Throwable {
   System.out.println("do something before method");
   Object ret = method.invoke(this.object, args);
   System.out.println("do something after method");
   return ret;
 
}
}

用戶端類:
 
package com.ml.test;
import java.lang.reflect.Proxy;
public class Client {
 
public static void main(String[] args) {
   // 元對象(被代理對象)
   ManagerImpl managerImpl = new ManagerImpl();
 
   // 業務代理類
   BusinessHandler securityHandler = new BusinessHandler(managerImpl);
 
   // 獲得代理類($Proxy0 extends Proxy implements Manager)的執行個體.
   Manager managerProxy = (Manager) Proxy.newProxyInstance(managerImpl
     .getClass().getClassLoader(), managerImpl.getClass()
     .getInterfaces(), securityHandler);
 
   managerProxy.modify();
}
}
三.執行結果:
do something before method
*******modify()方法被調用
do something after method
四.機制分析:
Proxy.(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)做了以下幾件事.
(1)根據參數loader和interfaces調用方法 getProxyClass(loader, interfaces)建立代理類$Proxy.
$Proxy0類實現了interfaces的介面,並繼承了Proxy類.
(2)執行個體化$Proxy0並在構造方法中把BusinessHandler傳過去,接著$Proxy0調用父類Proxy的構造器,為h賦值,如下:
class Proxy{
   InvocationHandler h=null;
   protected Proxy(InvocationHandler h) {
    this.h = h;
   }
   ...
}

 

下面是本例的$Proxy0類的源碼(好不容易才把它提出來):
 
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;
 
public final class $Proxy0 extends Proxy implements Manager {
 
private static Method m1;
private static Method m0;
private static Method m3;
private static Method m2;
 
static {
   try {
    m1 = Class.forName("java.lang.Object").getMethod("equals",
      new Class[] { Class.forName("java.lang.Object") });
    m0 = Class.forName("java.lang.Object").getMethod("hashCode",
      new Class[0]);
    m3 = Class.forName("com.ml.test.Manager").getMethod("modify",
      new Class[0]);
    m2 = Class.forName("java.lang.Object").getMethod("toString",
      new Class[0]);
   } catch (NoSuchMethodException nosuchmethodexception) {
    throw new NoSuchMethodError(nosuchmethodexception.getMessage());
   } catch (ClassNotFoundException classnotfoundexception) {
    throw new NoClassDefFoundError(classnotfoundexception.getMessage());
   }
}
 
public $Proxy0(InvocationHandler invocationhandler) {
   super(invocationhandler);
}
 
@Override
public final boolean equals(Object obj) {
   try {
    return ((Boolean) super.h.invoke(this, m1, new Object[] { obj }))
      .booleanValue();
   } catch (Throwable throwable) {
    throw new UndeclaredThrowableException(throwable);
   }
}
 
@Override
public final int hashCode() {
   try {
    return ((Integer) super.h.invoke(this, m0, null)).intValue();
   } catch (Throwable throwable) {
    throw new UndeclaredThrowableException(throwable);
   }
}
 
public final void modify() {
   try {
    super.h.invoke(this, m3, null);
    return;
   } catch (Error e) {
   } catch (Throwable throwable) {
    throw new UndeclaredThrowableException(throwable);
   }
}
 
@Override
public final String toString() {
   try {
    return (String) super.h.invoke(this, m2, null);
   } catch (Throwable throwable) {
    throw new UndeclaredThrowableException(throwable);
   }
}
}
接著把得到的$Proxy0執行個體強制轉換成Manager.
當執行managerProxy.modify()方法時,就調用了$Proxy0類中的modify()方法.
在modify方法中,調用父類Proxy中的h的invoke()方法.
即InvocationHandler.invoke();
以上一段是原作者的原話,在此本人詳細解說一下:
            
Manager managerProxy = (Manager) Proxy.newProxyInstance(managerImpl
     .getClass().getClassLoader(), managerImpl.getClass()
     .getInterfaces(), securityHandler);
 
此句中Proxy.newProxyInstance(..)方法執行時產生了$Proxy0的記憶體位元組碼檔案並return出來賦給了
managerProxy,強制轉化成了Manager介面,同時$Proxy0也實現了Manager介面中的所有方法,所以在
 managerProxy.modify(); 時就是調用了$Proxy0中的一下程式碼片段:
            
public final void modify() {
   try {
    super.h.invoke(this, m3, null); //該段則執行了InvocationHandler.invoke();  super.h既是InvocationHandler
    return;
   } catch (Error e) {
   } catch (Throwable throwable) {
    throw new UndeclaredThrowableException(throwable);
   }
}
 這樣動態代理機制就實現了。
 所以JAVA的動態代理的關鍵就在Proxy.newProxyInstance(..)方法執行時產生了$Proxy0的記憶體位元組碼以及JDK的反射機制!

作者“林清楊--技術部落格”

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.