Java躲不過(guò)設(shè)計(jì)模式的坑之代理模式詳解
前言
設(shè)計(jì)模式在我看來(lái)更像是一種設(shè)計(jì)思維或設(shè)計(jì)思想,它就像《孫子兵法》一樣,為你的項(xiàng)目工程提供方向,讓你的項(xiàng)目工程更加健壯、靈活,延續(xù)生命力。本文即將分享的是設(shè)計(jì)模式的其中一種:代理模式。
代理模式
通用官方定義:代理模式(Proxy Pattern) 是一種結(jié)構(gòu)型設(shè)計(jì)模式,通過(guò)代理對(duì)象控制對(duì)原對(duì)象的訪問(wèn),并允許在訪問(wèn)前或訪問(wèn)后做一些處理。
簡(jiǎn)單理解就是給一個(gè)對(duì)象找了一個(gè)替代品,這個(gè)替代品得到原對(duì)象授權(quán),可以攔截一些無(wú)效或低效的訪問(wèn),從而使得原對(duì)象可以釋放時(shí)間做自己的事情。這樣替代品實(shí)現(xiàn)了自己價(jià)值,原對(duì)象也得到了解放,兩全其美的選擇!!
代理主要分為以下幾種類型
- 靜態(tài)代理:僅用于單個(gè)接口實(shí)現(xiàn)類,程序運(yùn)行前已經(jīng)存在。調(diào)用時(shí)需要傳入具體實(shí)例,調(diào)用方可以直接獲取具體實(shí)例。
- 動(dòng)態(tài)代理:可以服務(wù)多個(gè)接口實(shí)現(xiàn)類,可以在程序運(yùn)行時(shí),通過(guò)反射機(jī)制動(dòng)態(tài)創(chuàng)建代理對(duì)象。
使用場(chǎng)景
既然這么說(shuō)了,那就結(jié)合實(shí)際介紹幾個(gè),還不是輕松拿捏~
場(chǎng)景一:作為一個(gè)氣血方剛的男青年,汽車總是繞不開(kāi)的話題,那就先以汽車為例。
4s店或汽車廠家均可以出售汽車,對(duì)于購(gòu)車消費(fèi)者來(lái)說(shuō),可以直接去喜歡的汽車店去體驗(yàn)成品,不必跋山涉水的跨省或跨市去汽車廠家。有了4S店代理,可以節(jié)省消費(fèi)者時(shí)間,更快體驗(yàn)到心儀的汽車,最終也是通過(guò)汽車廠家拿到成品;同時(shí)汽車工廠通過(guò)這些代理,可以更快售出汽車,可謂是一舉兩得。關(guān)系類圖如下:
場(chǎng)景二:上面講述的是靜態(tài)代理的案例,再來(lái)一個(gè)最近實(shí)踐的動(dòng)態(tài)代理需求場(chǎng)景。
人力業(yè)務(wù)平臺(tái)接入了不同客戶,同時(shí)為了滿足客戶不同的接入需求,采用動(dòng)態(tài)代理模式會(huì)為每個(gè)客戶動(dòng)態(tài)生成代理對(duì)象,比如需要提取客戶A的簡(jiǎn)歷數(shù)據(jù),根據(jù)客戶A的標(biāo)識(shí)獲取代理類并執(zhí)行對(duì)應(yīng)的實(shí)現(xiàn)邏輯,從而獲取客戶A的數(shù)據(jù)信息。關(guān)系如下
代碼分析
結(jié)合上述動(dòng)態(tài)代理業(yè)務(wù)場(chǎng)景,按照UML類型進(jìn)行代碼設(shè)計(jì),來(lái)演示下整體過(guò)程。首先將數(shù)據(jù)提取接口(IHandlerService)抽象出來(lái),同時(shí)提供一個(gè)通用實(shí)現(xiàn)(HandlerServiceImpl)。
/* * 通用接口 * */ public interface IHandlerService { /* * 抽取數(shù)據(jù) * */ void handle(); /** * 打印內(nèi)容 * * @param content 輸出內(nèi)容 */ String print(String content); /** * 設(shè)置信息 * * @param prefix 信息前綴 */ void setPrefix(String prefix); } /* * 通用實(shí)現(xiàn) * */ @Service public class HandlerServiceImpl implements IHandlerService { /* * 默認(rèn)信息前綴 * */ private String prefix = "default"; @Override public void handle() { System.out.println("=======自定義實(shí)現(xiàn)類" + prefix + "======"); } @Override public String print(String content) { System.out.println(prefix + " 實(shí)現(xiàn)類輸出 -》" + content); return prefix + "success"; } @Override public void setPrefix(String prefix) { this.prefix = prefix; } }
接下來(lái)創(chuàng)建個(gè)代理類,變量包含通用接口,也可以增加業(yè)務(wù)所需的其它變量。(java動(dòng)態(tài)代理核心內(nèi)容:InvocationHandler接口和Proxy類,代理對(duì)象在執(zhí)行函數(shù)時(shí),會(huì)通過(guò)InvocationHandler接口的invoke調(diào)用執(zhí)行函數(shù))具體代碼如下
public class MultiDynamicProxy implements InvocationHandler { /** * @see InvocationHandler * 每個(gè)代理實(shí)例的調(diào)用處理程序必須實(shí)現(xiàn)的接口,當(dāng)通過(guò)代理實(shí)例調(diào)用方法時(shí), * 這個(gè)方法的調(diào)用會(huì)被轉(zhuǎn)發(fā)至實(shí)現(xiàn)InvocationHandle接口類的invoke方法去執(zhí)行 */ private static Map<String, IHandlerService> map = new HashMap<>(); public static String key = "default"; private void addElements(String key) { if (map.containsKey(key)) return; IHandlerService handlerService = new HandlerServiceImpl(); handlerService.setPrefix(key); map.put(key, handlerService); } public static IHandlerService newInstance(IHandlerService handlerService) { MultiDynamicProxy handlerProxy = new MultiDynamicProxy(handlerService); // 抽象邏輯接口 Class<IHandlerService> handlerServiceClass = IHandlerService.class; /** * param1:指定接口(interface)的類加載器,用于裝入定義的代理類 * param2:動(dòng)態(tài)代理類要實(shí)現(xiàn)的接口 * param3:將執(zhí)行的代理方法調(diào)用派發(fā)給代理類(程序) * */ return (IHandlerService) Proxy.newProxyInstance(handlerServiceClass.getClassLoader(), new Class[]{handlerServiceClass}, handlerProxy); } public MultiDynamicProxy(IHandlerService handlerService) { map.put(key, handlerService); } /* * 自定義實(shí)現(xiàn)類對(duì)象替換代理類對(duì)象,并執(zhí)行 * param1:proxy 方法被調(diào)用的代理實(shí)例,即真實(shí)的代理對(duì)象 * param2:method 代理對(duì)象的method對(duì)象 * param3:args 代理對(duì)象方法傳遞的參數(shù) * */ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("=====代理類執(zhí)行====" + proxy.getClass().getSimpleName()); Object invoke = method.invoke(getElement(), args); return invoke; } private IHandlerService getElement() { if (null == map.get(key)) { addElements(key); } return map.get(key); }
小編在每個(gè)環(huán)節(jié)都增加了日志輸出,就很容易理解每個(gè)環(huán)節(jié)都做了哪些處理,日志內(nèi)容請(qǐng)接著往下看
public static void main(String[] args) { IHandlerService handlerService = MultiDynamicProxy.newInstance(new HandlerServiceImpl()); String s = handlerService.print("客戶A"); System.out.printf("執(zhí)行結(jié)果 => " + s); } // 輸出結(jié)果 // =====代理類執(zhí)行====$Proxy0 // default 實(shí)現(xiàn)類輸出 -》客戶A // 執(zhí)行結(jié)果 => defaultsuccess
觀察結(jié)果可以看出,執(zhí)行的service實(shí)例確實(shí)為代理對(duì)象($Proxy0),后續(xù)可以動(dòng)態(tài)接入客戶實(shí)現(xiàn),并注冊(cè)到客戶信息集合,當(dāng)然,也可以對(duì)實(shí)現(xiàn)類進(jìn)行擴(kuò)展,但考慮到通用性,所以接口職責(zé)盡可能保持單一,避免業(yè)務(wù)交叉,造成后續(xù)的維護(hù)困難。
總結(jié)
以上就是本文所分享的全部?jī)?nèi)容,靜態(tài)代理部分由于相對(duì)比較簡(jiǎn)單,就沒(méi)寫(xiě)代碼,主要是動(dòng)態(tài)代理,理解上比較容易,但是具體的執(zhí)行過(guò)程確實(shí)需要仔細(xì)分析,才能明白其實(shí)現(xiàn)原理。
代理模式確實(shí)對(duì)于目標(biāo)對(duì)象有保護(hù)作用,也方便了目標(biāo)對(duì)象的擴(kuò)展,但凡事都有兩面性,它也不是完美的,由于多了代理層,請(qǐng)求處理增加處理過(guò)程,進(jìn)而會(huì)降低響應(yīng)速度,同時(shí)也增加了系統(tǒng)復(fù)雜性,維護(hù)成本會(huì)有些增加。
沒(méi)有最完美的設(shè)計(jì)模式,只有最適合業(yè)務(wù)場(chǎng)景的設(shè)計(jì)模式。
以上就是Java躲不過(guò)設(shè)計(jì)模式的坑之代理模式詳解的詳細(xì)內(nèi)容,更多關(guān)于Java代理模式的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
java隨機(jī)生成一個(gè)名字和對(duì)應(yīng)拼音的方法
這篇文章主要介紹了java隨機(jī)生成一個(gè)名字和對(duì)應(yīng)拼音的方法,涉及java針對(duì)數(shù)組及隨機(jī)數(shù)操作的相關(guān)技巧,需要的朋友可以參考下2015-07-07Java多線程局域網(wǎng)聊天室的實(shí)現(xiàn)
在學(xué)習(xí)了一個(gè)學(xué)期的java以后,搞了一個(gè)多線程的聊天室,熟悉了一下服務(wù)器和客戶機(jī)的操作。感興趣的小伙伴們可以參考一下2021-06-06Java中MyBatis傳入?yún)?shù)parameterType問(wèn)題
這篇文章主要介紹了Java中MyBatis傳入?yún)?shù)parameterType問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12Spring JDBCTemplate原理及使用實(shí)例
這篇文章主要介紹了Spring JDBCTemplate原理及使用實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03Java動(dòng)態(tài)代理和AOP應(yīng)用示例
這篇文章主要介紹了Java動(dòng)態(tài)代理和AOP應(yīng)用,結(jié)合實(shí)例形式分析了java動(dòng)態(tài)代理在AOP面向切面編程中的相關(guān)操作技巧與使用注意事項(xiàng),需要的朋友可以參考下2019-07-07