設(shè)計模式之代理模式

代理模式的定義:代理模式給某一個對象提供一個代理對象怕享,并由代理對象控制對 原對象 的引用执赡。

? ? ? ? ?通俗的來講代理模式就是我們生活中常見的中介。

代理模式就是給一個對象提供一個代理函筋,并由代理對象控制對原對象的引用沙合。它使得客戶不能直接與真正的目標(biāo)對象通信。代理對象是目標(biāo)對象的代表跌帐,其他需要與這個目標(biāo)對象打交道的操作都是和這個代理對象在交涉首懈。

代理對象可以在客戶端和目標(biāo)對象之間起到中介的作用绊率,這樣起到了的作用和保護(hù)了目標(biāo)對象的,同時也在一定程度上面減少了系統(tǒng)的耦合度究履。



代理模式包含如下角色:

? ? ? Subject: 抽象主題角色

? ? ? Proxy: 代理主題角色

? ? ? ?RealSubject: 真實主題角色


為什么要用代理模式滤否?

中介隔離作用:在某些情況下,一個客戶類不想或者不能直接引用一個委托對象挎袜,而代理類對象可以在客戶類和委托對象之間起到中介的作用顽聂,其特征是代理類和委托類實現(xiàn)相同的接口肥惭。

開閉原則盯仪,增加功能:代理類除了是客戶類和委托類的中介之外,我們還可以通過給代理類增加額外的功能來擴(kuò)展委托類的功能蜜葱,這樣做我們只需要修改代理類而不需要再修改委托類全景,符合代碼設(shè)計的開閉原則。

代理類主要負(fù)責(zé)為委托類預(yù)處理消息牵囤、過濾消息爸黄、把消息轉(zhuǎn)發(fā)給委托類,以及事后對返回結(jié)果的處理等揭鳞。代理類本身并不真正實現(xiàn)服務(wù)炕贵,而是同過調(diào)用委托類的相關(guān)方法,來提供特定的服務(wù)野崇。真正的業(yè)務(wù)功能還是由委托類來實現(xiàn)称开,但是可以在業(yè)務(wù)功能執(zhí)行的前后加入一些公共的服務(wù)。例如我們想給項目加入緩存乓梨、日志這些功能鳖轰,我們就可以使用代理類來完成,而沒必要打開已經(jīng)封裝好的委托類扶镀。(例如AOP)

有哪幾種代理模式蕴侣?

? ? ? ?我們有多種不同的方式來實現(xiàn)代理。如果按照代理創(chuàng)建的時期來進(jìn)行分類的話臭觉, 可以分為兩種:靜態(tài)代理昆雀、動態(tài)代理。靜態(tài)代理是由程序員創(chuàng)建或特定工具自動生成源代碼蝠筑,在對其編譯狞膘。

? ? ? ? ? 靜態(tài)代理:在程序員運行之前,代理類.class文件就已經(jīng)被創(chuàng)建了菱肖。(IDE編譯等)

? ? ? ? ? ?動態(tài)代理:是在程序運行時通過反射機(jī)制動態(tài)創(chuàng)建的客冈。

1.靜態(tài)代理? ? ?

?一步:創(chuàng)建服務(wù)類接口

public interface BuyHouse {

? ? void buyHosue();

}


第二步:實現(xiàn)服務(wù)接口

public class BuyHouseImpl implements BuyHouse {

? ? @Override

? ? public void buyHosue() {

? ? ? ? System.out.println("我要買房");

? ? }

}

第三步:創(chuàng)建代理類

public class BuyHouseProxy implements BuyHouse {

? ? private BuyHouse buyHouse;

? ? public BuyHouseProxy(final BuyHouse buyHouse) {

? ? ? ? this.buyHouse = buyHouse;

? ? }

? ? @Override

? ? public void buyHosue() {

? ? ? ? System.out.println("買房前準(zhǔn)備");

? ? ? ? buyHouse.buyHosue();

? ? ? ? System.out.println("買房后裝修");

? ? }

}


第四步:編寫測試類

public class ProxyTest {

? ? public static void main(String[] args) {

? ? ? ? BuyHouse buyHouse = new BuyHouseImpl();

? ? ? ? buyHouse.buyHosue();

? ? ? ? BuyHouseProxy buyHouseProxy = new BuyHouseProxy(buyHouse);

? ? ? ? buyHouseProxy.buyHosue();

? ? }

}

靜態(tài)代理總結(jié):

優(yōu)點:可以做到在符合開閉原則的情況下對目標(biāo)對象進(jìn)行功能擴(kuò)展。

缺點:我們得為每一個服務(wù)都得創(chuàng)建代理類稳强,工作量太大场仲,不易管理和悦。同時接口一旦發(fā)生改變,代理類也得相應(yīng)修改渠缕。


? ?

2.動態(tài)代理

  在動態(tài)代理中我們不再需要再手動的創(chuàng)建代理類鸽素,我們只需要編寫一個動態(tài)處理器就可以了。真正的代理對象由JDK再運行時為我們動態(tài)的來創(chuàng)建亦鳞。

第一步:編寫動態(tài)處理器

public class DynamicProxyHandler implements InvocationHandler {

? ? private Object object;

? ? public DynamicProxyHandler(final Object object) {

? ? ? ? this.object = object;

? ? }

? ? @Override

? ? public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

? ? ? ? System.out.println("買房前準(zhǔn)備");

? ? ? ? Object result = method.invoke(object, args);

? ? ? ? System.out.println("買房后裝修");

? ? ? ? return result;

? ? }

}

第二步:編寫測試類

public class DynamicProxyTest {

? ? public static void main(String[] args) {

? ? ? ? BuyHouse buyHouse = new BuyHouseImpl();

? ? ? ? BuyHouse proxyBuyHouse = (BuyHouse) Proxy.newProxyInstance(BuyHouse.class.getClassLoader(), new

? ? ? ? ? ? ? ? Class[]{BuyHouse.class}, new DynamicProxyHandler(buyHouse));

? ? ? ? proxyBuyHouse.buyHosue();

? ? }

}

注意Proxy.newProxyInstance()方法接受三個參數(shù):

ClassLoader loader:指定當(dāng)前目標(biāo)對象使用的類加載器,獲取加載器的方法是固定的

Class<?>[] interfaces:指定目標(biāo)對象實現(xiàn)的接口的類型,使用泛型方式確認(rèn)類型

InvocationHandler:指定動態(tài)處理器馍忽,執(zhí)行目標(biāo)對象的方法時,會觸發(fā)事件處理器的方法

動態(tài)代理總結(jié):雖然相對于靜態(tài)代理,動態(tài)代理大大減少了我們的開發(fā)任務(wù)燕差,同時減少了對業(yè)務(wù)接口的依賴遭笋,降低了耦合度。但是還是有一點點小小的遺憾之處徒探,那就是它始終無法擺脫僅支持interface代理的桎梏瓦呼,因為它的設(shè)計注定了這個遺憾。


3.CGLIB代理

JDK實現(xiàn)動態(tài)代理需要實現(xiàn)類通過接口定義業(yè)務(wù)方法测暗,對于沒有接口的類央串,如何實現(xiàn)動態(tài)代理呢,這就需要CGLib了碗啄。CGLib采用了非常底層的字節(jié)碼技術(shù)质和,其原理是通過字節(jié)碼技術(shù)為一個類創(chuàng)建子類,并在子類中采用方法攔截的技術(shù)攔截所有父類方法的調(diào)用稚字,順勢織入橫切邏輯饲宿。但因為采用的是繼承,所以不能對final修飾的類進(jìn)行代理尉共。

JDK動態(tài)代理與CGLib動態(tài)代理均是實現(xiàn)Spring AOP的基礎(chǔ)褒傅。

第一步:創(chuàng)建CGLIB代理類

public class CglibProxy implements MethodInterceptor {

? ? private Object target;

? ? public Object getInstance(final Object target) {

? ? ? ? this.target = target;

? ? ? ? Enhancer enhancer = new Enhancer();

? ? ? ? enhancer.setSuperclass(this.target.getClass());

? ? ? ? enhancer.setCallback(this);

? ? ? ? return enhancer.create();

? ? }

? ? public Object intercept(Object object, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {

? ? ? ? System.out.println("買房前準(zhǔn)備");

? ? ? ? Object result = methodProxy.invoke(object, args);

? ? ? ? System.out.println("買房后裝修");

? ? ? ? return result;

? ? }

}

第二步:創(chuàng)建測試類

public class CglibProxyTest {

? ? public static void main(String[] args){

? ? ? ? BuyHouse buyHouse = new BuyHouseImpl();

? ? ? ? CglibProxy cglibProxy = new CglibProxy();

? ? ? ? BuyHouseImpl buyHouseCglibProxy = (BuyHouseImpl) cglibProxy.getInstance(buyHouse);

? ? ? ? buyHouseCglibProxy.buyHosue();

? ? }

}

CGLIB代理總結(jié):?CGLIB創(chuàng)建的動態(tài)代理對象比JDK創(chuàng)建的動態(tài)代理對象的性能更高,但是CGLIB創(chuàng)建代理對象時所花費的時間卻比JDK多得多袄友。所以對于單例的對象殿托,因為無需頻繁創(chuàng)建對象,用CGLIB合適剧蚣,反之使用JDK方式要更為合適一些支竹。同時由于CGLib由于是采用動態(tài)創(chuàng)建子類的方法,對于final修飾的方法無法進(jìn)行代理鸠按。





更多詳細(xì)礼搁,請點擊查看

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市目尖,隨后出現(xiàn)的幾起案子馒吴,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,589評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件饮戳,死亡現(xiàn)場離奇詭異豪治,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)扯罐,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,615評論 3 396
  • 文/潘曉璐 我一進(jìn)店門负拟,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人歹河,你說我怎么就攤上這事掩浙。” “怎么了秸歧?”我有些...
    開封第一講書人閱讀 165,933評論 0 356
  • 文/不壞的土叔 我叫張陵厨姚,是天一觀的道長。 經(jīng)常有香客問我寥茫,道長遣蚀,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,976評論 1 295
  • 正文 為了忘掉前任纱耻,我火速辦了婚禮,結(jié)果婚禮上险耀,老公的妹妹穿的比我還像新娘弄喘。我一直安慰自己,他們只是感情好甩牺,可當(dāng)我...
    茶點故事閱讀 67,999評論 6 393
  • 文/花漫 我一把揭開白布蘑志。 她就那樣靜靜地躺著,像睡著了一般贬派。 火紅的嫁衣襯著肌膚如雪急但。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,775評論 1 307
  • 那天搞乏,我揣著相機(jī)與錄音波桩,去河邊找鬼。 笑死请敦,一個胖子當(dāng)著我的面吹牛镐躲,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播侍筛,決...
    沈念sama閱讀 40,474評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼萤皂,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了匣椰?” 一聲冷哼從身側(cè)響起裆熙,我...
    開封第一講書人閱讀 39,359評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后入录,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體齐媒,經(jīng)...
    沈念sama閱讀 45,854評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,007評論 3 338
  • 正文 我和宋清朗相戀三年纷跛,在試婚紗的時候發(fā)現(xiàn)自己被綠了喻括。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,146評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡贫奠,死狀恐怖唬血,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情唤崭,我是刑警寧澤拷恨,帶...
    沈念sama閱讀 35,826評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站谢肾,受9級特大地震影響腕侄,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜芦疏,卻給世界環(huán)境...
    茶點故事閱讀 41,484評論 3 331
  • 文/蒙蒙 一冕杠、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧酸茴,春花似錦分预、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,029評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至酪穿,卻和暖如春凳干,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背被济。 一陣腳步聲響...
    開封第一講書人閱讀 33,153評論 1 272
  • 我被黑心中介騙來泰國打工救赐, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人溉潭。 一個月前我還...
    沈念sama閱讀 48,420評論 3 373
  • 正文 我出身青樓净响,卻偏偏與公主長得像,于是被迫代替她去往敵國和親喳瓣。 傳聞我的和親對象是個殘疾皇子馋贤,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,107評論 2 356

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