日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

InvocationHandler的invoke方法如何被调用?

發布時間:2024/2/28 编程问答 26 豆豆
生活随笔 收集整理的這篇文章主要介紹了 InvocationHandler的invoke方法如何被调用? 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

關鍵問題是它們內在的聯系。雖然可以看源代碼,包括類庫。但是把它們抽象出來,有助于理解。

?

在客戶類(即以下代碼的Client類),語句Manager managerProxy =

??(Manager) Proxy.newProxyInstance(managerImpl.getClass().getClassLoader(), managerImpl.getClass().getInterfaces(), securityHandler);

?

??可以看到,Proxy(庫類)中的newProxyInstance方法被調用,該方法返回一個被代理對象的實例,然后向上轉型為其對應的接口。

?

??問題是該方法在返回之前已經做了什么?我們看看其參數的第三個:securityHandler,它就是InvocationHandler接口實現類的一個實例:securityHandler。因為InvocationHandler接口中原本有個invoke的方法,所以其實現類當然需要實現這個方法,即其實現類(在此是BusinessHandler類)中有invoke方法,而如果invoke方法要被調用,只能通過BusinessHandler類對象來調用。

?

??而在下面的源代碼類$Proxy0?中,有三個地方已經用BusinessHandler類對象來調用invoke方法,他們分別在這些方法的代碼中:public final boolean equals(Object obj) ;public final int hashCode() ;public final String toString()。也許你會有疑問,這個方法也沒有看到在哪里被

$Proxy0 對象調用過,怎么能執行invoke方法呢?但是請看$Proxy0中的static代碼塊,這個模塊是特殊的,因為當newProxyInstance創建$Proxy0 時,它就被初始化。而這個static模塊中的getMethod方法加載了這個三個方法,因而它們里面的代碼(h.invoke())被執行。

?

??另外我們還可以看到,invoke方法中的第一個參數代表什么。在源代碼中我們看到return ((Boolean) super.h.invoke(this, m1, new Object[] { obj })) .booleanValue(); 那么this就是$Proxy0 對象。這里需要注意的是,它并非是被代理對象(ManagerImpl)。其第二個參數傳遞的是$Proxy0 類中數據域,其類型是Method,是被封裝過的被代理對象(ManagerImpl)的方法。

??小結:invoke方法的調用過程,就是先新建其類對象(實例),然后把它傳入newProxyInstance方法中,在里面解析并用它來調用invoke方法。

?

?

一.相關類及其方法:

java.lang.reflect.Proxy類的newProxyInstance(),是用于創建動態代理類和實例的靜態方法.返回一個指定接口的代理類實例,該接口可以將方法調用指派到指定的調用處理程序。

java.lang.reflect.InvocationHandler接口中的invoke(),在代理實例上處理方法調用并返回結果。當與方法關聯的代理實例上調用方法時,將在調用處理程序上調用此方法。

二.探討的源代碼:

被代理對象的接口及實現類:

public interface Manager {
public void modify();
}

?

public class ManagerImpl implements Manager {
??public void modify() {
?????System.out.println("*******modify()方法被調用");
??}
}

?

動態代理類:

?

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;

???}
}



客戶端類:
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.newProxyInstance(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());
???}
} //static


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()方法時(managerProxy是Manager的一個已經向上轉型的實例),就調用了$Proxy0類中的modify()方法.


???在modify方法中,調用父類Proxy中的h的invoke()方法.即InvocationHandler.invoke();

總結

以上是生活随笔為你收集整理的InvocationHandler的invoke方法如何被调用?的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。