Java躲不過設(shè)計(jì)模式的坑之代理模式詳解
前言
設(shè)計(jì)模式在我看來更像是一種設(shè)計(jì)思維或設(shè)計(jì)思想,它就像《孫子兵法》一樣,為你的項(xiàng)目工程提供方向,讓你的項(xiàng)目工程更加健壯、靈活,延續(xù)生命力。本文即將分享的是設(shè)計(jì)模式的其中一種:代理模式。
代理模式
通用官方定義:代理模式(Proxy Pattern) 是一種結(jié)構(gòu)型設(shè)計(jì)模式,通過代理對(duì)象控制對(duì)原對(duì)象的訪問,并允許在訪問前或訪問后做一些處理。
簡單理解就是給一個(gè)對(duì)象找了一個(gè)替代品,這個(gè)替代品得到原對(duì)象授權(quá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í),通過反射機(jī)制動(dòng)態(tài)創(chuàng)建代理對(duì)象。
使用場(chǎng)景
既然這么說了,那就結(jié)合實(shí)際介紹幾個(gè),還不是輕松拿捏~
場(chǎng)景一:作為一個(gè)氣血方剛的男青年,汽車總是繞不開的話題,那就先以汽車為例。
4s店或汽車廠家均可以出售汽車,對(duì)于購車消費(fèi)者來說,可以直接去喜歡的汽車店去體驗(yàn)成品,不必跋山涉水的跨省或跨市去汽車廠家。有了4S店代理,可以節(jié)省消費(fèi)者時(shí)間,更快體驗(yàn)到心儀的汽車,最終也是通過汽車廠家拿到成品;同時(shí)汽車工廠通過這些代理,可以更快售出汽車,可謂是一舉兩得。關(guān)系類圖如下:

場(chǎng)景二:上面講述的是靜態(tà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的簡歷數(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ì),來演示下整體過程。首先將數(shù)據(jù)提取接口(IHandlerService)抽象出來,同時(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;
}
}接下來創(chuàng)建個(gè)代理類,變量包含通用接口,也可以增加業(yè)務(wù)所需的其它變量。(java動(dòng)態(tài)代理核心內(nèi)容:InvocationHandler接口和Proxy類,代理對(duì)象在執(zhí)行函數(shù)時(shí),會(huì)通過InvocationHandler接口的invoke調(diào)用執(zhí)行函數(shù))具體代碼如下
public class MultiDynamicProxy implements InvocationHandler {
/**
* @see InvocationHandler
* 每個(gè)代理實(shí)例的調(diào)用處理程序必須實(shí)現(xiàn)的接口,當(dāng)通過代理實(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é)
以上就是本文所分享的全部內(nèi)容,靜態(tài)代理部分由于相對(duì)比較簡單,就沒寫代碼,主要是動(dòng)態(tài)代理,理解上比較容易,但是具體的執(zhí)行過程確實(shí)需要仔細(xì)分析,才能明白其實(shí)現(xiàn)原理。
代理模式確實(shí)對(duì)于目標(biāo)對(duì)象有保護(hù)作用,也方便了目標(biāo)對(duì)象的擴(kuò)展,但凡事都有兩面性,它也不是完美的,由于多了代理層,請(qǐng)求處理增加處理過程,進(jìn)而會(huì)降低響應(yīng)速度,同時(shí)也增加了系統(tǒng)復(fù)雜性,維護(hù)成本會(huì)有些增加。
沒有最完美的設(shè)計(jì)模式,只有最適合業(yè)務(wù)場(chǎng)景的設(shè)計(jì)模式。
以上就是Java躲不過設(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-07
Java多線程局域網(wǎng)聊天室的實(shí)現(xiàn)
在學(xué)習(xí)了一個(gè)學(xué)期的java以后,搞了一個(gè)多線程的聊天室,熟悉了一下服務(wù)器和客戶機(jī)的操作。感興趣的小伙伴們可以參考一下2021-06-06
Java中MyBatis傳入?yún)?shù)parameterType問題
這篇文章主要介紹了Java中MyBatis傳入?yún)?shù)parameterType問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12
Spring JDBCTemplate原理及使用實(shí)例
這篇文章主要介紹了Spring JDBCTemplate原理及使用實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03
Java動(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

