淺談設(shè)計(jì)模式之代理模式

代理模式基本概念

定義:

代理模式(Proxy Pattern)為其他對(duì)象提供一種代理以控制對(duì)這個(gè)對(duì)象的訪問赠幕。代理對(duì)象起到中介作用俄精,可去掉功能服務(wù)或增加額外的服務(wù)。

類圖:

代理模式

根據(jù)以上類圖榕堰,可以知道在代理模式中的角色有以下三種:

  1. 抽象對(duì)象角色:聲明了目標(biāo)對(duì)象和代理對(duì)象的共同接口竖慧,這樣一來在任何可以使用目標(biāo)對(duì)象的地方都可以使用代理對(duì)象。

  2. 目標(biāo)對(duì)象角色:定義了代理對(duì)象所代表的目標(biāo)對(duì)象逆屡。

  3. 代理對(duì)象角色:代理對(duì)象內(nèi)部含有目標(biāo)對(duì)象的引用圾旨,從而可以在任何時(shí)候操作目標(biāo)對(duì)象;代理對(duì)象提供一個(gè)與目標(biāo)對(duì)象相同的接口魏蔗,以便可以在任何時(shí)候替代目標(biāo)對(duì)象砍的。代理對(duì)象通常在客戶端調(diào)用傳遞給目標(biāo)對(duì)象之前或之后,執(zhí)行某個(gè)操作莺治,而不是單純地將調(diào)用傳遞給目標(biāo)對(duì)象廓鞠。

分類:

  1. 遠(yuǎn)程代理(Remote Proxy):為不同地理的對(duì)象提供局域網(wǎng)代表對(duì)象
  2. 虛擬代理(Virtual Proxy):根據(jù)需要將資源消耗很大的對(duì)象進(jìn)行延遲,真正需要的時(shí)候再創(chuàng)建
  3. 保護(hù)代理(Protect Proxy):控制用戶的訪問權(quán)限
  4. 智能引用代理(Smart Reference Proxy):提供對(duì)目標(biāo)對(duì)象額外的服務(wù)

在使用代理模式的時(shí)候谣旁,九成以上使用到的都是智能引用代理床佳,故以下案例都圍繞智能代理展開。

常用代理模式原理

代理模式實(shí)現(xiàn)的兩種方式:

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

首先我們先來看看靜態(tài)代理的實(shí)現(xiàn)原理榄审。


靜態(tài)代理(Static Proxy)

定義:

代理和被代理對(duì)象在代理之前都是確定的砌们。他們都實(shí)現(xiàn)相同的接口或者繼承相同的抽象類

靜態(tài)代理實(shí)現(xiàn)方法:

  • 繼承法:代理類直接繼承被代理類搁进,實(shí)現(xiàn)其原有方法浪感,并添加一些額外功能
  • 聚合法:代理類實(shí)現(xiàn)相同的功能接口(很重要,相同接口饼问,不同代理也可以進(jìn)行相互代理)篮撑,并在內(nèi)聲明一個(gè)被代理類的對(duì)象(類似封裝),通過內(nèi)部對(duì)象實(shí)現(xiàn)其原有方法匆瓜,并添加額外功能

靜態(tài)代理實(shí)現(xiàn)代碼

首先定義一個(gè)接口,這里我們來拿汽車的行駛舉例。

public interface Moveable {
    void move();
}

然后弄一個(gè)車驮吱,實(shí)現(xiàn)這個(gè)接口茧妒。Car這個(gè)類就是即將要被代理的對(duì)象。

import java.util.Random;

public class Car implements Moveable {

    @Override
    public void move() {
        //實(shí)現(xiàn)開車
        try {
            Thread.sleep(new Random().nextInt(1000));
            System.out.println("汽車行駛中....");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

}

先來看看繼承法實(shí)現(xiàn)靜態(tài)代理左冬。

public class Car2 extends Car {

    @Override
    public void move() {
        long starttime = System.currentTimeMillis();
        System.out.println("汽車開始行駛....");
        super.move();
        long endtime = System.currentTimeMillis();
        System.out.println("汽車結(jié)束行駛....  汽車行駛時(shí)間:" + (endtime - starttime) + "毫秒桐筏!");
    }
}

再來看看聚合法實(shí)現(xiàn)靜態(tài)代理。

public class Car3 implements Moveable {

    public Car3(Car car) {
        super();
        this.car = car;
    }

    private Car car;
    
    @Override
    public void move() {
        long starttime = System.currentTimeMillis();
        System.out.println("汽車開始行駛....");
        car.move();
        long endtime = System.currentTimeMillis();
        System.out.println("汽車結(jié)束行駛....  汽車行駛時(shí)間:" + (endtime - starttime) + "毫秒拇砰!");
    }

}

最后提供一個(gè)客戶端的測(cè)試類梅忌。

public class Client {

    public static void main(String[] args) {
        //使用繼承方式實(shí)現(xiàn)靜態(tài)代理
        Moveable m1 = new Car2();
        m1.move();
        //使用聚合方式實(shí)現(xiàn)靜態(tài)代理
        Car car = new Car();
        Moveable m2 = new Car3(car);
        m2.move();
    }

}

那么繼承法和聚合法何者更優(yōu)呢?

結(jié)論是聚合比繼承更適合實(shí)現(xiàn)代理除破。

如果我們要實(shí)現(xiàn)功能的疊加牧氮,比如增加方法運(yùn)行時(shí)間處理,增加權(quán)限管理瑰枫,增加日志處理等踱葛,或者調(diào)整這些功能的實(shí)現(xiàn)順序,如果有一百種實(shí)現(xiàn)順序光坝,我們用繼承法就要實(shí)現(xiàn)100個(gè)代理子類尸诽,這顯然是不現(xiàn)實(shí)的,如果需要修改也是極其繁瑣的盯另。而如果使用聚合法性含,只要一個(gè)代理類管理一個(gè)功能,在維護(hù)和調(diào)整順序上都是最優(yōu)的鸳惯。這里就不再舉例了商蕴。


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

類圖:

首先看一下動(dòng)態(tài)代理的類圖。


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

定義:

所謂Dynamic Proxy是這樣一種class:

它是在運(yùn)行時(shí)生成的class悲敷,該class需要實(shí)現(xiàn)一組interface究恤,使用動(dòng)態(tài)代理類時(shí),必須實(shí)現(xiàn)InvocationHandle接口后德。

實(shí)現(xiàn)步驟:

  1. 創(chuàng)建一個(gè)實(shí)現(xiàn)接口InvocationHandler的類部宿,它必須實(shí)現(xiàn)invoke方法
  2. 創(chuàng)建被代理的類以及接口
  3. 調(diào)用Proxy的靜態(tài)方法(newProxyInstance(ClassLoader loader,Class[] interfaces,InvocationHandler h)),創(chuàng)建一個(gè)代理類
  4. 通過代理調(diào)用方法

動(dòng)態(tài)代理實(shí)現(xiàn)方法:

  • JDK動(dòng)態(tài)產(chǎn)生代理
  • cglib動(dòng)態(tài)產(chǎn)生代理

動(dòng)態(tài)代理實(shí)現(xiàn)代碼:

JDK動(dòng)態(tài)代理:
public interface Moveable {
    void move();
}
import java.util.Random;

public class Car implements Moveable {

    @Override
    public void move() {
        //實(shí)現(xiàn)開車
        try {
            Thread.sleep(new Random().nextInt(1000));
            System.out.println("汽車行駛中....");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

}

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

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class TimeHandler implements InvocationHandler {

    public TimeHandler(Object target) {
        super();
        this.target = target;
    }

    private Object target;
    
    /*
     * 參數(shù):
     * proxy  被代理對(duì)象
     * method  被代理對(duì)象的方法
     * args 方法的參數(shù)
     * 
     * 返回值:
     * Object  方法的返回值
     * */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        long starttime = System.currentTimeMillis();
        System.out.println("汽車開始行駛....");
        method.invoke(target);
        long endtime = System.currentTimeMillis();
        System.out.println("汽車結(jié)束行駛....  汽車行駛時(shí)間:" 
                + (endtime - starttime) + "毫秒瓢湃!");
        return null;
    }

}

測(cè)試類:

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;

public class Test {

    /**
     * JDK動(dòng)態(tài)代理測(cè)試類
     */
    public static void main(String[] args) {
        Car car = new Car();
        InvocationHandler h = new TimeHandler(car);
        Class<?> cls = car.getClass();
        /**
         * loader  類加載器
         * interfaces  實(shí)現(xiàn)接口
         * h InvocationHandler
         */
        Moveable m = (Moveable)Proxy.newProxyInstance(cls.getClassLoader(),cls.getInterfaces(), h);
        m.move();
    }
}

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

首先需要導(dǎo)入第三方j(luò)ar包——cglib-nodep-2.2.jar

創(chuàng)建一個(gè)火車類理张。

public class Train {

    public void move(){
        System.out.println("火車行駛中...");
    }
}

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

import java.lang.reflect.Method;

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

public class CglibProxy implements MethodInterceptor {

    private Enhancer enhancer = new Enhancer();
    
    public Object getProxy(Class clazz){
        //設(shè)置創(chuàng)建子類的類
        enhancer.setSuperclass(clazz);
        enhancer.setCallback(this);
        
        return enhancer.create();
    }
    
    /**
     * 攔截所有目標(biāo)類方法的調(diào)用
     * obj  目標(biāo)類的實(shí)例
     * m   目標(biāo)方法的反射對(duì)象
     * args  方法的參數(shù)
     * proxy代理類的實(shí)例
     */
    @Override
    public Object intercept(Object obj, Method m, Object[] args,MethodProxy proxy) throws Throwable {
        System.out.println("日志開始...");
        //代理類調(diào)用父類的方法
        proxy.invokeSuper(obj, args);
        System.out.println("日志結(jié)束...");
        return null;
    }

}

測(cè)試類:

public class Client {

    /**
     * @param args
     */
    public static void main(String[] args) {

        CglibProxy proxy = new CglibProxy();
        Train t = (Train)proxy.getProxy(Train.class);
        t.move();
    }

}

參考資料:

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市绵患,隨后出現(xiàn)的幾起案子雾叭,更是在濱河造成了極大的恐慌,老刑警劉巖落蝙,帶你破解...
    沈念sama閱讀 206,839評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件织狐,死亡現(xiàn)場(chǎng)離奇詭異暂幼,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)移迫,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門旺嬉,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人厨埋,你說我怎么就攤上這事邪媳。” “怎么了荡陷?”我有些...
    開封第一講書人閱讀 153,116評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵雨效,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我废赞,道長(zhǎng)徽龟,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,371評(píng)論 1 279
  • 正文 為了忘掉前任蛹头,我火速辦了婚禮顿肺,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘渣蜗。我一直安慰自己屠尊,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,384評(píng)論 5 374
  • 文/花漫 我一把揭開白布耕拷。 她就那樣靜靜地躺著讼昆,像睡著了一般。 火紅的嫁衣襯著肌膚如雪骚烧。 梳的紋絲不亂的頭發(fā)上浸赫,一...
    開封第一講書人閱讀 49,111評(píng)論 1 285
  • 那天,我揣著相機(jī)與錄音赃绊,去河邊找鬼既峡。 笑死,一個(gè)胖子當(dāng)著我的面吹牛碧查,可吹牛的內(nèi)容都是我干的运敢。 我是一名探鬼主播,決...
    沈念sama閱讀 38,416評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼忠售,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼传惠!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起稻扬,我...
    開封第一講書人閱讀 37,053評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤卦方,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后泰佳,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體盼砍,經(jīng)...
    沈念sama閱讀 43,558評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡尘吗,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,007評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了浇坐。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片摇予。...
    茶點(diǎn)故事閱讀 38,117評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖吗跋,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情宁昭,我是刑警寧澤跌宛,帶...
    沈念sama閱讀 33,756評(píng)論 4 324
  • 正文 年R本政府宣布,位于F島的核電站积仗,受9級(jí)特大地震影響疆拘,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜寂曹,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,324評(píng)論 3 307
  • 文/蒙蒙 一哎迄、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧隆圆,春花似錦漱挚、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,315評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至侣背,卻和暖如春白华,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背贩耐。 一陣腳步聲響...
    開封第一講書人閱讀 31,539評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工弧腥, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人潮太。 一個(gè)月前我還...
    沈念sama閱讀 45,578評(píng)論 2 355
  • 正文 我出身青樓管搪,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親消别。 傳聞我的和親對(duì)象是個(gè)殘疾皇子抛蚤,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,877評(píng)論 2 345

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