一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|JAVA教程|ASP教程|

服務器之家 - 編程語言 - JAVA教程 - java動態代理詳解

java動態代理詳解

2019-11-11 13:44java教程網 JAVA教程

本文章要用很土的語言描述Java動態代理,力求更易被理解。Java是JDK5中新加的機制,大家都知道Spring是用Java的動態代理實現的,那這個動態代理是什么東東呢,首先他肯定是個代理,我們先講代理,把代理弄明白了,動態代理就好

代理都知道吧,你去買東西就有很多的代理商,他們就是賣原廠的東西。比如,你天天要買肉,豬是農民伯伯養的,但你是從屠夫手上買到肉的,這個屠夫就可以當成是代理。那為什么要代理呢,代理有什么用呢,當然是有事給他做了,對于屠夫這個代理就好理解了,因為你自己不可能去宰豬吧,所以代理就是去買活豬,然后宰掉再賣給你,當然屠夫有可能給肉注點水,關鍵看他壞不壞,所以屠夫的整個流程就是:

這個流程用代碼怎么實現呢:我們應該要用三個類You、Butcher、Farmer分別指你、屠夫、農民伯伯。其中農民伯伯又提供一個買肉的方法給屠夫調用,這個方法輸入是錢的數量,返回是肉的數量,都用int型,代碼如下:

 

復制代碼代碼如下:

class Farmer {
    public int buyMeat(int money) {
        int meat = 0;
        // ... meat = ***;
        return meat;
    }
}

 

而屠夫則提供一個買肉的方法給你調用,同樣是輸入錢,返回肉,但是會把肉加工一下(殺豬和刮豬毛在代碼中就省了,要不然還得為豬寫個類),代碼如下:

 

復制代碼代碼如下:

class Butcher {
    public int buyMeat(int money) {
        Farmer farmer = new Farmer();            // 1.find a farmer.
        int meat = farmer.buyMeat(money);        // 2.buy meat from the farmer.
        meat += 5;                               // 3.inject 5 pound water into the meat, so weight will increase. 
        return meat;                             // 4.return to you.
    }
}

 

然你從屠夫手上買肉的代碼就變成這樣:

 

復制代碼代碼如下:

class You {
    public void work() {
        int youMoney = 10;
        Butcher butcher = new Butcher();        // find a butcher.
        int meat = butcher.buyMeat(youMoney);
        System.out.println("Cook the meat, weight: " + meat);  // you cooked it. 
    }
}

 

這個程序我們還可以優化一下,我們發現屠夫有農民有一個相同的買肉方法,我們可以提取一個接口,叫為商販(pedlar)吧,以后你買肉就不用管他是屠夫還是農民伯伯了,只要他有肉賣就可以了,我們提取一個接口后,代碼就變成這樣:

 

復制代碼代碼如下:


class You {
    public void work() {
        int youMoney = 10;
        Peldar peldar= new Butcher();                               // find a peldar.
        int meat = peldar.buyMeat(youMoney);
        System.out.println("Cook the meat, weight: " + meat);        // you cooked it.    
    }
}
interface Peldar {
 int buyMeat(int money);
}
class Butcher implements Peldar {
    @Override
    public int buyMeat(int money) {
        Farmer farmer = new Farmer();            // 1.find a farmer.
        int meat = farmer.buyMeat(money);        // 2.buy meat from the farmer.
        meat += 5;                               // 3.inject 5 pound water into the meat, so weight will increase. 
        return meat;                             // 4.return to you.
    }
}

 

class Farmer implements Peldar {
    @Override
    public int buyMeat(int money) {
        int meat = 0;
        // ... meat = ***;
        return meat;
    }
}

 

這就是代理,值得注意的是一般代理類和最終類會實現同一接口,這樣的好處是,調用者就不用關心當前引用的到底是代理還是最終類。

不過這叫靜態代理,因為代理類(屠夫類)是你親手寫,動態代理就是Java在運行的時候,動態生成一個等價的代理類。雖然類是動態生成的,但是殺豬和注水的代碼還是要寫的,只是不要寫一個類了。寫到哪里呢,寫到下面這個接口里面:

 

復制代碼代碼如下:

public interface InvocationHandler { 
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable; 
}

 

參數是什么意思呢,我寫成這樣,可能你就明白了:

 

復制代碼代碼如下:

public interface InvocationHandler { 
    public Object invoke(Object butcher, Method buyMeat, Object[] money) throws Throwable; 
}

 

第一個參數是自動生成的代理類的一個對象(自動生成的屠夫類的對象),第二個參數是正前正在被調用的方法的對象(方法怎么還有對象呢,參見Java反射機制),我們這里只有一個方法叫buyMeat,所以這個參數代表的肯定就是它了,第三個參數是傳給前面那個方法的參數數組,buyMeat只有一個參數,所以這個數組只會有一個元素。于是殺豬注水的代碼寫進來就變成這樣了:

 

復制代碼代碼如下:

InvocationHandler mInvocationHandler = new InvocationHandler() {  
    @Override
    public Object invoke(Object butcher, Method buyMeat, Object[] args) throws Throwable {
        Farmer farmer = new Farmer();              // 1.find a farmer.
        int meat = (Integer) buyMeat.invoke(farmer, args);      // 2.buy meat from the farmer.
        meat += 5;                                 // 3.inject 5 pound water into the meat, so weight will increase. 
        return meat;                               // 4.return to you.
    }
};

 

這個里調用農民伯伯的買肉方法有點不符常規,這里是反射機制調用法,意思是這樣的,以farmer對象為接受者來調用buyMeat方法,跟直接調用farmer的方法是一樣的,你可能會問那為什么不直接調用呢,你可能沒注意,invoke的第一個參數類型是Object,所以你可以向任何對象發布調用命令(但不一定會成功,什么時候會成功等下說),如果你有很多farmer對象,甚至不是farmer對象,只要某接口的實例就可以(哪個接口等下說明,我們先命名為A接口),就可以當成參數傳進來,然后對其進行方法調用。現在我們來看看如何生成代理類吧,很簡單,可以調用Proxy的工廠方法,如下:

 

復制代碼代碼如下:

public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)  
    throws IllegalArgumentException

 

解釋參數,第一個ClassLoader是用來加載代理類的(關于ClassLoader,本文暫不講解),你暫不了解也沒關系,第二個是一個數組,每個元數都是一個接口,新生成的代理都會實現所有這些接口,傳給InvocationHandler.invoke第二個參數的方法,必定屬于所有這些接口中的方法,上一段落說的那個A接口必須是數組中的一個元素,上一段落說的那個調用成失敗問題也明了了。第三個參數InvocationHandler更好理解了,就是只要代理類中的任何方法被調用,就會通知這個InvocationHandler。下面寫出完整代碼:

 

復制代碼代碼如下:


class You {
    public void work() {
        int youMoney = 10;

        Peldar peldarProxy = (Peldar) Proxy.newProxyInstance(getClass().getClassLoader(), new Class[]{Peldar.class}, mInvocationHandler);
        int meat = peldarProxy.buyMeat(youMoney);

        System.out.println("Cook the meat, weight: " + meat);    
    }

    InvocationHandler mInvocationHandler = new InvocationHandler() {        
        @Override
        public Object invoke(Object butcher, Method buyMeat, Object[] args)
                throws Throwable {
            Farmer farmer = new Farmer();                           // 1.find a farmer.
            int meat = (Integer) buyMeat.invoke(farmer, args);      // 2.buy meat from the farmer.
            meat += 5;                                              // 3.inject 5 pound water into the meat, so weight will increase. 
            return meat;                                            // 4.return to you.
        }
    };

}
interface Peldar {
    int buyMeat(int money);
}

 

class Farmer implements Peldar {
    @Override
    public int buyMeat(int money) {
        int meat = 0;
        // ... meat = ***;
        return meat;
    }
}

 

這里You類里生成一個代理類,在代理類的buyMeat被調用時,代碼就跟之前的靜態代理一樣的了。

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 欧美一区二区日韩一区二区 | 操碰91 | 成年女人毛片免费观看97 | 精品久久久久久久久久久 | 四虎永久免费地址在线观看 | 亚洲一区二区三区福利在线 | 吉川爱美与黑人解禁 | 亚洲男人天堂a | 久操久操久操 | 亚洲色图首页 | 15同性同志18 | 国产精品人人视频 | 好涨好大我快受不了了视频网 | 午夜伦伦电影理论片费看 | 成年人视频在线 | 人人澡人 | 日韩日b视频 | 情缘1完整版在线观看 | 国产精品第1页在线播放 | 别停好爽好深好大好舒服视频 | 97色伦| 日韩去日本高清在线 | 青草网址 | 99re视频精品全部免费 | 精品视频一区二区 | 精品淑女少妇AV久久免费 | 国产普通话对白露脸流出 | 日本阿v精品视频在线观看 日本xxx片免费高清在线 | 日韩精品视频在线观看免费 | 青草精品| 国产愉拍精品视频手机 | 98精品全国免费观看视频 | 国产在线精品亚洲第一区香蕉 | 欧美疯狂做爰3xxx | 俄罗斯三级在线观看级 | swag最新正在播放 | 欧美高清在线不卡免费观看 | 91久| 免费看黄色片的网站 | 免费高清视频在线观看 | 国产成人在线免费观看 |