揭秘JDK動(dòng)態(tài)代理

代理在我們?nèi)粘i_(kāi)發(fā)過(guò)程中有著很重要的角色驴一,它可以處理一些日志記錄休雌,權(quán)限控制,事務(wù)等蛔趴。主要分為靜態(tài)代理和動(dòng)態(tài)代理挑辆,他們的主要區(qū)別是有沒(méi)有源文件,靜態(tài)代理是有源文件的孝情,應(yīng)用起來(lái)比較直觀但是稍有浪費(fèi)且不智能鱼蝉,所以我們一般使用的是動(dòng)態(tài)代理,他是在程序運(yùn)行期間動(dòng)態(tài)生成的箫荡,程序加載動(dòng)態(tài)生成的字節(jié)碼生成Class對(duì)象加以運(yùn)用魁亦,靈活性高。
下面我們介紹一下jdk的動(dòng)態(tài)代理羔挡。
首先聲明一個(gè)接口IHello.java

public interface IHello {
    void sayHello(String name);
}

在創(chuàng)建一個(gè)接口的具體的實(shí)現(xiàn)類HelloService.java

public class HelloService implements IHello {
    @Override
    public void sayHello(String name) {
        System.out.println("hello "+name);
    }
}

使用jdk的動(dòng)態(tài)代理洁奈,代理的邏輯必須寫在InvocationHandler的實(shí)現(xiàn)類中间唉,當(dāng)調(diào)用接口的具體方法時(shí),實(shí)際上是調(diào)用的InvocationHandler中的invoke方法利术,這個(gè)在后面會(huì)有詳細(xì)的解釋呈野。
HelloHandler.java

public class HelloHandler implements InvocationHandler {
    /**
     * 調(diào)用的具體的對(duì)象
     */
    private Object concreteObj;

    public HelloHandler(IHello hello) {
        concreteObj = hello;
    }

    /**
     * @param proxy   生成的代理對(duì)象的回填
     * @param method
     * @param args
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("proxy:"+proxy.getClass().getName());
        long start = System.currentTimeMillis();
        Object o = method.invoke(concreteObj, args);
        System.out.println("cost time :" + (System.currentTimeMillis() - start) + "ms");
        return o;
    }
}

需要的類都準(zhǔn)備好了,我們來(lái)測(cè)試一下

    public static void main(String[] args) {
        IHello hello = (IHello) Proxy.newProxyInstance(IHello.class.getClassLoader(), new Class[]{IHello.class}, new HelloHandler(new HelloService()));
        hello.sayHello("test");
    }

結(jié)果為:

proxy:com.sun.proxy.$Proxy0
hello test
cost time :0ms

我們這個(gè)動(dòng)態(tài)代理的作用就是統(tǒng)計(jì)一下方法調(diào)用的時(shí)間印叁。
我們發(fā)現(xiàn)了兩個(gè)問(wèn)題:

  1. 這個(gè)HelloHandler中invoke方法第一個(gè)參數(shù)proxy是怎么填入的并且他的class的name為何為com.sun.proxy.$Proxy0
  2. 我調(diào)用的是IHello中的sayHello方法被冒,為啥他會(huì)走到HelloHandler中的invoke方法。

我們帶著這兩個(gè)疑問(wèn)繼續(xù)看起來(lái)轮蜕。
我們先看一下Proxy是如何newProxyInstance的

110.png

下面我們看一下生成的字節(jié)碼是怎樣的

    public static void main(String[] args) {
         createProxyClassFile();
    }

    public static void createProxyClassFile() {
        String name = "ProxyHello";
        byte[] data = ProxyGenerator.generateProxyClass(name, new Class[]{IHello.class});
        try {
            FileOutputStream out = new FileOutputStream(name + ".class");
            out.write(data);
            out.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

創(chuàng)建的文件這這樣的

public final class ProxyHello extends Proxy implements IHello {
    private static Method m1;
    private static Method m3;
    private static Method m2;
    private static Method m0;

    public ProxyHello(InvocationHandler var1) throws  {
        super(var1);
    }

    public final boolean equals(Object var1) throws  {
        try {
            return ((Boolean)super.h.invoke(this, m1, new Object[]{var1})).booleanValue();
        } catch (RuntimeException | Error var3) {
            throw var3;
        } catch (Throwable var4) {
            throw new UndeclaredThrowableException(var4);
        }
    }

    public final void sayHello(String var1) throws  {
        try {
            super.h.invoke(this, m3, new Object[]{var1});
        } catch (RuntimeException | Error var3) {
            throw var3;
        } catch (Throwable var4) {
            throw new UndeclaredThrowableException(var4);
        }
    }

    public final String toString() throws  {
        try {
            return (String)super.h.invoke(this, m2, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }

    public final int hashCode() throws  {
        try {
            return ((Integer)super.h.invoke(this, m0, (Object[])null)).intValue();
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }

    static {
        try {
            m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[]{Class.forName("java.lang.Object")});
            m3 = Class.forName("com.pajk.homecare.unittest.proxy.IHello").getMethod("sayHello", new Class[]{Class.forName("java.lang.String")});
            m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
            m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
        } catch (NoSuchMethodException var2) {
            throw new NoSuchMethodError(var2.getMessage());
        } catch (ClassNotFoundException var3) {
            throw new NoClassDefFoundError(var3.getMessage());
        }
    }
}

根據(jù)這個(gè)文件昨悼,我們看一下,當(dāng)調(diào)用代理的sayHello方法時(shí)跃洛,調(diào)用的是handler的invoke方法率触,傳入的第一個(gè)參數(shù)是this,所以就是Proxy的實(shí)例了汇竭。所以上面提出的兩個(gè)問(wèn)題得解葱蝗。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市韩玩,隨后出現(xiàn)的幾起案子垒玲,更是在濱河造成了極大的恐慌,老刑警劉巖找颓,帶你破解...
    沈念sama閱讀 219,589評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異叮贩,居然都是意外死亡击狮,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,615評(píng)論 3 396
  • 文/潘曉璐 我一進(jìn)店門益老,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)彪蓬,“玉大人,你說(shuō)我怎么就攤上這事捺萌〉刀” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 165,933評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵桃纯,是天一觀的道長(zhǎng)酷誓。 經(jīng)常有香客問(wèn)我,道長(zhǎng)态坦,這世上最難降的妖魔是什么盐数? 我笑而不...
    開(kāi)封第一講書人閱讀 58,976評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮伞梯,結(jié)果婚禮上玫氢,老公的妹妹穿的比我還像新娘帚屉。我一直安慰自己,他們只是感情好漾峡,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,999評(píng)論 6 393
  • 文/花漫 我一把揭開(kāi)白布攻旦。 她就那樣靜靜地躺著,像睡著了一般生逸。 火紅的嫁衣襯著肌膚如雪敬特。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書人閱讀 51,775評(píng)論 1 307
  • 那天牺陶,我揣著相機(jī)與錄音伟阔,去河邊找鬼。 笑死掰伸,一個(gè)胖子當(dāng)著我的面吹牛皱炉,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播狮鸭,決...
    沈念sama閱讀 40,474評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼合搅,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了歧蕉?” 一聲冷哼從身側(cè)響起灾部,我...
    開(kāi)封第一講書人閱讀 39,359評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎惯退,沒(méi)想到半個(gè)月后赌髓,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,854評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡催跪,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,007評(píng)論 3 338
  • 正文 我和宋清朗相戀三年锁蠕,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片懊蒸。...
    茶點(diǎn)故事閱讀 40,146評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡荣倾,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出骑丸,到底是詐尸還是另有隱情舌仍,我是刑警寧澤,帶...
    沈念sama閱讀 35,826評(píng)論 5 346
  • 正文 年R本政府宣布通危,位于F島的核電站铸豁,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏黄鳍。R本人自食惡果不足惜推姻,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,484評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望框沟。 院中可真熱鬧藏古,春花似錦增炭、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 32,029評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至厂捞,卻和暖如春输玷,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背靡馁。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 33,153評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工欲鹏, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人臭墨。 一個(gè)月前我還...
    沈念sama閱讀 48,420評(píng)論 3 373
  • 正文 我出身青樓赔嚎,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親胧弛。 傳聞我的和親對(duì)象是個(gè)殘疾皇子尤误,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,107評(píng)論 2 356

推薦閱讀更多精彩內(nèi)容