Java靜態(tài)代理&動(dòng)態(tài)代理筆記

java代理

閱讀原文請(qǐng)?jiān)L問(wèn)我的博客BrightLoong's Blog

最近在學(xué)習(xí)Java反射的一些知識(shí),看到了一些有關(guān)代理的例子,好記性不如爛筆頭灰殴,所以這里將它記錄下來(lái)。接下來(lái)話不多說(shuō)掰邢,直接進(jìn)入主題牺陶。

代理:為其他對(duì)象提供一個(gè)代理以控制對(duì)某個(gè)對(duì)象的訪問(wèn)。

靜態(tài)代理

  • 接口
public interface IDoSomething {
    public int doSometing(int num);
}
  • 被代理類的實(shí)現(xiàn)
public class Sing implements IDoSomething {

    @Override
    public int doSometing(int num) {
        System.out.println("Sing a song");
        return num;
    }
}
  • 代理類的實(shí)現(xiàn)
 public class SingProxy implements IDoSomething{
    
    private IDoSomething sing = new Sing();

    @Override
    public int doSometing(int num) {
        System.out.println("Befor singing ");
        int result = sing.doSometing(num);
        System.out.println("After singing");
        return result;
    }

}
  • 測(cè)試類
public class ProxyDemo {
     public static int sing(IDoSomething sing, int num) {
         return sing.doSometing(num);
     }
     public static void main(String[] args) {
         System.out.println(ProxyDemo.sing(new SingProxy(), 5));
     }
 }
  • 輸出結(jié)果
Befor singing 
Sing a song
After singing
5

以上就是簡(jiǎn)單的靜態(tài)代理尸变,不在過(guò)多的介紹义图,下面是動(dòng)態(tài)代理。

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

Java的動(dòng)態(tài)代理可以動(dòng)態(tài)的創(chuàng)建代理并動(dòng)態(tài)的處理對(duì)所代理方法的調(diào)用召烂。動(dòng)態(tài)代理有兩種實(shí)現(xiàn)方法碱工,一種是使用JDK自帶的,一種是使用Cglib實(shí)現(xiàn)奏夫。

實(shí)現(xiàn)JDK自帶的動(dòng)態(tài)代理

實(shí)現(xiàn)JDK自帶的動(dòng)態(tài)代理怕篷,關(guān)鍵是實(shí)現(xiàn)InvocationHandler,同時(shí)它要求被代理對(duì)象必須有接口酗昼。下面是實(shí)現(xiàn)的代碼廊谓,我加上了必要的注釋。

  • 接口
public interface IProxyClass {
    public int doSomething(int i);
}
  • 被代理類的實(shí)現(xiàn)
public class ProxyClassImpl implements IProxyClass {
    @Override
    public int doSomething(int num) {
        System.out.println("方法執(zhí)行中.....");
        return num;
    }
}
  • 實(shí)現(xiàn)InvocationHandler接口
    ??這里我實(shí)現(xiàn)了InvocationHandler接口麻削,并手動(dòng)生成了代理類蒸痹,保存到了電腦F盤上
public class DynamicProxyHandler implements InvocationHandler {
    
    private Object proxied;
    
    /**
     * @param proxied 被代理對(duì)象
     */
    public DynamicProxyHandler(Object proxied) {
        this.proxied = proxied;
    }
    
    /**
     * 返回代理對(duì)象
     * @return
     */
    public Object newProxyInstance() {
        return Proxy.newProxyInstance(proxied.getClass().getClassLoader(), proxied.getClass().getInterfaces(), this);
    }

    /**
     * 
     * @param proxy 代理對(duì)象
     * @param method 代理方法
     * @param args 方法參數(shù)
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { 
        //將代理對(duì)象生成字節(jié)碼到F盤上,方便反編譯出java文件查看呛哟,實(shí)際動(dòng)態(tài)代理是不需要自己生成的
        addClassToDisk(proxy.getClass().getName(), ProxyClassImpl.class,"F:/$Proxy0.class");
        System.out.println("method:"+method.getName());  
        System.out.println("args:"+args[0].getClass().getName());  
        System.out.println("Before invoke method...");  
        Object object=method.invoke(proxied, args);
        System.out.println("After invoke method...");  
        return object;  
    }
    
    /**
     * 用于生產(chǎn)代理對(duì)象的字節(jié)碼叠荠,并將其保存到硬盤上
     * @param className
     * @param cl
     * @param path
     */
    private void addClassToDisk(String className, Class<?> cl, String path) {
        //用于生產(chǎn)代理對(duì)象的字節(jié)碼
        byte[] classFile = ProxyGenerator.generateProxyClass(className, cl.getInterfaces());
        FileOutputStream out = null;  
        try {  
            out = new FileOutputStream(path);  
            //將代理對(duì)象的class字節(jié)碼寫到硬盤上
            out.write(classFile);  
            out.flush();  
        } catch (Exception e) {  
            e.printStackTrace();  
        } finally {  
            try {  
                out.close();  
            } catch (IOException e) {  
                e.printStackTrace();  
            }  
        }  
    }
    
}
  • 測(cè)試類
public class SimpleProxyDemo {
      public static void main(String[] args) throws SecurityException, NoSuchMethodException {
          ProxyClassImpl c = new ProxyClassImpl();
          DynamicProxyHandler proxyHandler = new DynamicProxyHandler(c);
          IProxyClass proxyClass = (IProxyClass)proxyHandler.newProxyInstance();
          System.out.println(proxyClass.getClass().getName());
          System.out.println(proxyClass.doSomething(5));
      }
}
  • 輸出結(jié)果
com.sun.proxy.$Proxy0
method:doSomething
args:java.lang.Integer
Before invoke method...
方法執(zhí)行中.....
After invoke method...
5

從結(jié)果我們可以看到(IProxyClass)proxyHandler.newProxyInstance();實(shí)際返回的是com.sun.proxy.$Proxy0,我們把生成的$Proxy0.class文件扫责,使用jad.exe進(jìn)行反編譯榛鼎,使用命令(要求文件和jad.exe在同一個(gè)目錄下,或者你可以吧jad加到環(huán)境變量中去):

jad -p java $Proxy0.class

得到的$Proxy0.java如下:

public final class $Proxy0 extends Proxy
    implements IProxyClass
{

    public $Proxy0(InvocationHandler invocationhandler)
    {
        super(invocationhandler);
    }

    public final boolean equals(Object obj)
    {
        try
        {
            return ((Boolean)super.h.invoke(this, m1, new Object[] {
                obj
            })).booleanValue();
        }
        catch(Error _ex) { }
        catch(Throwable throwable)
        {
            throw new UndeclaredThrowableException(throwable);
        }
    }

    public final void doSomething(int i)
    {
        try
        {
            super.h.invoke(this, m3, new Object[] {
                Integer.valueOf(i)
            });
            return;
        }
        catch(Error _ex) { }
        catch(Throwable throwable)
        {
            throw new UndeclaredThrowableException(throwable);
        }
    }

    public final int hashCode()
    {
        try
        {
            return ((Integer)super.h.invoke(this, m0, null)).intValue();
        }
        catch(Error _ex) { }
        catch(Throwable throwable)
        {
            throw new UndeclaredThrowableException(throwable);
        }
    }

    public final String toString()
    {
        try
        {
            return (String)super.h.invoke(this, m2, null);
        }
        catch(Error _ex) { }
        catch(Throwable throwable)
        {
            throw new UndeclaredThrowableException(throwable);
        }
    }

    private static Method m1;
    private static Method m3;
    private static Method m0;
    private static Method m2;

    static 
    {
        try
        {
            m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[] {
                Class.forName("java.lang.Object")
            });
            m3 = Class.forName("io.github.brightloong.proxy.IProxyClass").getMethod("doSomething", new Class[] {
                Integer.TYPE
            });
            m0 = Class.forName("java.lang.Object").getMethod("hashCode", 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());
        }
    }
}

可以看到實(shí)際調(diào)用的是25行的doSometing()方法。如果你想了解更加具體的JDK動(dòng)態(tài)代理的實(shí)現(xiàn)原理可以訪問(wèn)Rejoy的博文JDK動(dòng)態(tài)代理實(shí)現(xiàn)原理者娱。

使用Cglib實(shí)現(xiàn)動(dòng)態(tài)代理

Cglib不是jdk自帶的jar包抡笼,需要下載并加入到項(xiàng)目中。個(gè)人覺(jué)得Cglib比使用jdk自帶的實(shí)現(xiàn)動(dòng)態(tài)代理更為先進(jìn)黄鳍,畢竟它不再需要接口推姻,而且它還有其他強(qiáng)大的功能,大家可以自行研究际起。

  • 實(shí)現(xiàn)MethodInterceptor接口
public class CglibProxy implements MethodInterceptor{
    private Enhancer enhancer = new Enhancer();
    @Override
    /**
     * 
     * @param o 是被代理對(duì)象
     * @param method 調(diào)用方法的Method對(duì)象
     * @param args 方法參數(shù)
     * @param methodProxy
     * @return cglib生成用來(lái)代替Method對(duì)象的一個(gè)對(duì)象拾碌,使用MethodProxy比調(diào)用JDK自身的Method直接執(zhí)行方法效率會(huì)有提升
     * @throws Throwable
     */
    public Object intercept(Object o, Method method, Object[] args,
            MethodProxy methodProxy) throws Throwable {
        System.out.println("before " + methodProxy.getSuperName());  
        System.out.println(method.getName());  
        Object o1 = methodProxy.invokeSuper(o, args);  
        //Object o2 = method.invoke(o, args); 使用這種方式會(huì)發(fā)生死循環(huán),因?yàn)榉椒〞?huì)被攔截
        System.out.println("after " + methodProxy.getSuperName());  
        return o1;  
    }
    
    public  Object newProxyInstance(Class<?> c) {
        //設(shè)置產(chǎn)生的代理對(duì)象的父類街望。
        enhancer.setSuperclass(c); 
        //設(shè)置CallBack接口的實(shí)例
        enhancer.setCallback(this);  
        //使用默認(rèn)無(wú)參數(shù)的構(gòu)造函數(shù)創(chuàng)建目標(biāo)對(duì)象 
        return enhancer.create();  
    }
}
  • 被代理對(duì)象和測(cè)試類
public class CglibDemo {
    public static void main(String[] args) {
        CglibProxy cglibProxy = new CglibProxy();  
        Do o = (Do)cglibProxy.newProxyInstance(Do.class);  
        System.out.println(o.doSomething(5));
    }
}
class Do{
    public int doSomething(int num){
        System.out.println("方法執(zhí)行中校翔。。灾前。防症。。哎甲。");
        return num;
    }
}
  • 輸出結(jié)果
before CGLIB$doSomething$0
doSomething
方法執(zhí)行中蔫敲。。炭玫。奈嘿。。吞加。
after CGLIB$doSomething$0
5

本篇筆記參考于:

http://www.cnblogs.com/shijiaqi1066/p/3429691.html

http://rejoy.iteye.com/blog/1627405

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末裙犹,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子衔憨,更是在濱河造成了極大的恐慌叶圃,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,194評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件践图,死亡現(xiàn)場(chǎng)離奇詭異掺冠,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)码党,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門德崭,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人揖盘,你說(shuō)我怎么就攤上這事眉厨。” “怎么了扣讼?”我有些...
    開封第一講書人閱讀 156,780評(píng)論 0 346
  • 文/不壞的土叔 我叫張陵缺猛,是天一觀的道長(zhǎng)缨叫。 經(jīng)常有香客問(wèn)我椭符,道長(zhǎng)荔燎,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,388評(píng)論 1 283
  • 正文 為了忘掉前任销钝,我火速辦了婚禮有咨,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘蒸健。我一直安慰自己座享,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,430評(píng)論 5 384
  • 文/花漫 我一把揭開白布似忧。 她就那樣靜靜地躺著渣叛,像睡著了一般。 火紅的嫁衣襯著肌膚如雪盯捌。 梳的紋絲不亂的頭發(fā)上淳衙,一...
    開封第一講書人閱讀 49,764評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音饺著,去河邊找鬼箫攀。 笑死,一個(gè)胖子當(dāng)著我的面吹牛幼衰,可吹牛的內(nèi)容都是我干的靴跛。 我是一名探鬼主播,決...
    沈念sama閱讀 38,907評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼渡嚣,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼梢睛!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起严拒,我...
    開封第一講書人閱讀 37,679評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤扬绪,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后裤唠,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體挤牛,經(jīng)...
    沈念sama閱讀 44,122評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,459評(píng)論 2 325
  • 正文 我和宋清朗相戀三年种蘸,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了墓赴。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,605評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡航瞭,死狀恐怖诫硕,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情刊侯,我是刑警寧澤章办,帶...
    沈念sama閱讀 34,270評(píng)論 4 329
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響藕届,放射性物質(zhì)發(fā)生泄漏挪蹭。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,867評(píng)論 3 312
  • 文/蒙蒙 一休偶、第九天 我趴在偏房一處隱蔽的房頂上張望梁厉。 院中可真熱鬧,春花似錦踏兜、人聲如沸词顾。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,734評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)肉盹。三九已至,卻和暖如春疹尾,著一層夾襖步出監(jiān)牢的瞬間垮媒,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,961評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工航棱, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留睡雇,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,297評(píng)論 2 360
  • 正文 我出身青樓饮醇,卻偏偏與公主長(zhǎng)得像它抱,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子朴艰,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,472評(píng)論 2 348

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

  • 從三月份找實(shí)習(xí)到現(xiàn)在观蓄,面了一些公司,掛了不少祠墅,但最終還是拿到小米侮穿、百度、阿里毁嗦、京東亲茅、新浪、CVTE狗准、樂(lè)視家的研發(fā)崗...
    時(shí)芥藍(lán)閱讀 42,209評(píng)論 11 349
  • 轉(zhuǎn)自:http://xiezhaodong.me/2017/03/31/Java-JDK%E4%BB%A3%E7%...
    王帥199207閱讀 1,268評(píng)論 1 17
  • 隨著一年辛勞的結(jié)束,2017新年的鐘聲也敲響了捞附。 年的味道越來(lái)越濃巾乳。不知是自己敏感還是怎么您没,今年這年過(guò)得真冷清。 ...
    莫念zyj閱讀 282評(píng)論 0 0
  • 文 /淺漸深 ‘ 與你相遇胆绊,好幸運(yùn)紊婉! ’ 楚源曾說(shuō): 在愛(ài)情里, 我就是個(gè)拿的起放不下的人 辑舷,誰(shuí)遇...
    淺漸深閱讀 327評(píng)論 0 0
  • 以前是只想著每天都寫點(diǎn)什么東西,但是每次槽片,手機(jī)打上幾個(gè)字是何缓,打了刪刪了打,其實(shí)到底我也不知道該說(shuō)些什么。直到用了這...
    wangaq閱讀 204評(píng)論 0 0