日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
詳解設(shè)計模式之代理模式

設(shè)計模式已經(jīng)跟大家分享很多了常見的模式了,感興趣的小伙伴可以再回顧一下,鞏固一下理解。

這次要跟大家分享的是設(shè)計模式中三大類創(chuàng)建型中的代理模式,代理模式在業(yè)務(wù)場景上我們可能不會經(jīng)常用到,但是面試官卻會經(jīng)常問一個問題?

請你跟我講講Spring里面AOP的代理模式?jdk的代理模式和cglib的代理模式又啥區(qū)別?

清楚和不清楚的同學都可以接著向下看,一定會有收獲。

言歸正傳,接下來開始一步一步分析一下代理模式。

定義以及目的

首先代理模式可以分為多種類型

  • 遠程代理:就是將工作委托給遠程對象(不同的服務(wù)器,或者不同的進程)來完成。常見的是用在web Service中。還有就是我們的RPC調(diào)用也可以理解為一種遠程代理。
  • 保護代理:該模式主要進行安全/權(quán)限檢查。(接觸很少)
  • 緩存代理:這個很好理解,就是通過存儲來加速調(diào)用,比如Sping中的@Cacheable方法,緩存特定的參數(shù)獲取到的結(jié)果,當下次相同參數(shù)調(diào)用該方法,直接從緩存中返回數(shù)據(jù)。
  • 虛擬代理:這種代理主要是為方法增加功能,比如記錄一些性能指標等,或進行延遲初始化

上面只是我們作為了解的概念,接下來再看看代理模式有哪些部分構(gòu)成。

  • Subject(共同接口):客戶端使用的現(xiàn)有接口
  • RealSubject(真實對象):真實對象的類
  • ProxySubject(代理對象):代理類

從圖中可以看出其實整個接口還是很簡單,就是一個真實對象以及代理對象。

  • 目的:提供一個實際代理對象,以便更好的控制實際對象。以上定義來自《設(shè)計模式之美》

代碼舉例實現(xiàn)

為了方便理解,還是舉一個例子,不知道大家在讀初中或者高中是否經(jīng)歷過傳小紙條的過程,假如現(xiàn)在同學A 對同學C有一些話想聊(比如放學相約一起打游戲)但是因為現(xiàn)在是上課時間,又不能大聲說,同學A和同學C之間坐了一個同學B,所以現(xiàn)在同學A只能是先找到同學B把紙條給它,讓他轉(zhuǎn)告同學C,但是去玩還是不是不去玩,那還是只能真正的同學C自己才能決定。

所以代理模式可以理解為 同學B是同學的C的代理,同學A要找同學C,只能找到同學B,通過同學B轉(zhuǎn)達同學C,同時將同學的C的執(zhí)行結(jié)果反饋給同學A。

說完了例子還是具體看看代碼的實現(xiàn)吧

 
 
 
 
  1. public interface Subject {
  2.    // 共同的接口
  3.     void doSomething();
  4. }

定義一個共同的接口(即大家要做的事請:放學一起打游戲)

 
 
 
 
  1. public class RealSubject implements Subject {
  2.    // 真實對象
  3.     @Override
  4.     public void doSomething() {
  5.         System.out.println("放學去打游戲");
  6.     }
  7. }

構(gòu)建一個真實對象,即例子中的同學C

 
 
 
 
  1. public class ProxySubject implements Subject {
  2.     private RealSubject realSubject;
  3.     public ProxySubject(RealSubject realSubject) {
  4.         this.realSubject = realSubject;
  5.     }
  6.     public ProxySubject() throws ClassNotFoundException, IllegalAccessException, InstantiationException {
  7.         this.realSubject = (RealSubject) this.getClass().getClassLoader().loadClass("com.ao.bing.demo.proxyPattern.RealSubject").newInstance();
  8.     }
  9.     @Override
  10.     public void doSomething() {
  11.         realSubject.doSomething();
  12.     }
  13.     public static void main(String[] args) {
  14.         try {
  15.             // 第一種方式
  16.             new ProxySubject().doSomething();
  17.             // 打印結(jié)果: 放學去打游戲
  18.         } catch (Exception e) {
  19.             // 異常情況,代理失敗,
  20.             // 傳紙條的被老師抓了。或者同學C不在座位上了 等異常情況
  21.         }
  22.         // 第二種方式
  23.         new ProxySubject(new RealSubject()).doSomething();
  24.         // 打印結(jié)果: 放學去打游戲
  25.     }
  26. }

構(gòu)建代理對象,即同學B,那么可以看到同學A并沒有真實接觸到同學C,通過同學B對同學C的代理就能知道同學C放學能不能跟他一起去打游戲

在Main方法里面,有兩種方式來調(diào)用真實對象

  • 第一種:采用類加載器形式,去加載實列對象,這樣我們就不同關(guān)心到底什么時候需要真實的實列化對象
  • 第二種:通過傳值的形式,把實列化對象傳過來。(理解為裝飾器模式了)

這里大家要區(qū)別一下,代理模式是提供完全相同的接口,而裝飾器模式是為了增強接口。

靜態(tài)代理、動態(tài)代理和cglib代理分析

靜態(tài)代理

在上面的舉的列子實現(xiàn)其實就是靜態(tài)代理,大家可以看到整體也比較簡單。但是它的缺點也很明顯

靜態(tài)代理需要為每一個對象都創(chuàng)建一個代理類,增加了維護成本以及開發(fā)成本,那么為了解決這個問題,動態(tài)代理就出來了,不要再固定為每一個需要代理的類而創(chuàng)建一個代理類

動態(tài)代理

動態(tài)代理合理的避免了靜態(tài)代理的那種方式,不用事先為要代理的類而構(gòu)建好代理類。而是在運行時通過反射機制創(chuàng)建。

在寫動態(tài)代理事需要理解兩個東西:Proxy 可以理解為就是調(diào)度器,InvocationHandler 增強服務(wù)接口可以理解為代理器。所以我個人理解動態(tài)代理其實就是一種行為的監(jiān)聽。

具體的代碼實現(xiàn)舉一個例子:螳螂捕蟬,通過通過螳螂監(jiān)聽到蟬的動作。方便后面講到多級代理模式。

 
 
 
 
  1. public interface BaseService {
  2.     void mainService();
  3. }
  4. public class Cicada implements BaseService {
  5.     @Override
  6.     public void mainService() {
  7.         System.out.println("主要業(yè)務(wù),以蟬為例,當蟬出現(xiàn)業(yè)務(wù)調(diào)用時,螳螂監(jiān)聽到");
  8.     }
  9. }

創(chuàng)建共同接口,以及真實對象蟬

 
 
 
 
  1. public class PrayingMantis implements InvocationHandler {
  2.     private BaseService baseService;
  3.   // 這里采用的是構(gòu)建傳參數(shù),可以用反射,舉的第一個例子有樣式代碼
  4.     public PrayingMantis(BaseService baseService) {
  5.         this.baseService = baseService;
  6.     }
  7.     // 螳螂主要業(yè)務(wù),也就是監(jiān)聽對象
  8.     @Override
  9.     public Object invoke(Object listener, Method method, Object[] args) throws Throwable {
  10.         method.invoke(baseService,args);
  11.         secondaryMain();
  12.         return null;
  13.     }
  14.     // 這里理解增強業(yè)務(wù),即我們可以在實現(xiàn)InvocationHandler里面添加其他的業(yè)務(wù),比如日志等等。
  15.     private void secondaryMain(){
  16.         System.out.println("螳螂捕蟬 - 次要業(yè)務(wù)");
  17.     }
  18. }

創(chuàng)建螳螂類,監(jiān)聽著蟬的類的動作

 
 
 
 
  1. public class BeanFactory {
  2.     public static BaseService newInstanc(Class classFile) {
  3.         // 1. 創(chuàng)建蟬,真實類對象
  4.         BaseService trueCicada = new Cicada();
  5.         // 2.創(chuàng)建代理類 螳螂
  6.         InvocationHandler prayingMantis = new PrayingMantis(trueCicada);
  7.         // 3.向Jvm索要代理對象 其實就是監(jiān)聽的對象,
  8.         Class classArray[] = {BaseService.class};
  9.         BaseService baseService = (BaseService) Proxy.newProxyInstance(classFile.getClassLoader(), classArray, prayingMantis);
  10.         return baseService;
  11.     }
  12.   
  13.    // 測試Demo
  14.     public static void main(String[] args) {
  15.         BaseService baseService  = newInstanc(Cicada.class);
  16.         baseService.mainService();
  17.         // 測試結(jié)果 :主要業(yè)務(wù)
  18.         //           螳螂捕蟬 - 次要業(yè)務(wù)
  19.     }
  20. }

通過結(jié)果可以看出當蟬主要業(yè)務(wù)發(fā)生調(diào)用時,螳螂能監(jiān)聽到蟬的業(yè)務(wù)并且能處理其他業(yè)務(wù)邏輯,這也就是Spring里面AOP為什么能處理日志切面等。

代理的本質(zhì):

  • 我認為其實就是一種行為的監(jiān)聽,對代理對象($proxy InvocationHandler)的一種監(jiān)聽行為。

代理模式組成:

  • 接口:聲明需要被監(jiān)聽行為
  • 代理實現(xiàn)類(InvocationHandler):次要業(yè)務(wù),次要業(yè)務(wù)和主要業(yè)務(wù)綁定執(zhí)行
  • 代理對象(監(jiān)聽對象)
  • Cglib動態(tài)代理

cglib動態(tài)代理

其實和jdk的動態(tài)代理是很相似的,都是要去實現(xiàn)代理器接口完成。

具體代碼如下:

 
 
 
 
  1. public class PrayingMantis implements MethodInterceptor {
  2.     private Cicada cicada;// 代理對象
  3.     public Cicada getInstance(Cicada cicada) {
  4.         this.cicada = cicada;
  5.         Enhancer enhancer = new Enhancer();
  6.         enhancer.setSuperclass(this.cicada.getClass());
  7.         enhancer.setCallback(this);
  8.         return (Cicada) enhancer.create();
  9.     }
  10.     @Override
  11.     public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
  12.         Object object = methodProxy.invokeSuper(o, objects);
  13.         secondaryMain();
  14.         return object;
  15.     }
  16.     private void secondaryMain() {
  17.         System.out.println("螳螂捕蟬 - 次要業(yè)務(wù)");
  18.     }
  19.     public static void main(String[] args) {
  20.         PrayingMantis prayingMantis = new PrayingMantis();
  21.         Cicada instance = prayingMantis.getInstance(new Cicada());
  22.         instance.mainService();
  23.         // 結(jié)果:主要業(yè)務(wù)
  24.         //      螳螂捕蟬 - 次要業(yè)務(wù)
  25.     }

因為蟬類都是一樣的所以我就不單獨這里再貼出來。

細心的同學已經(jīng)發(fā)現(xiàn),Cglib 無需通過接口來實現(xiàn),它是通過實現(xiàn)子類的方式來完成調(diào)用的。

  • Enhancer 對象把代理對象設(shè)置為被代理類的子類來實現(xiàn)動態(tài)代理的。因為是采用繼承方式,所以代理類不能加final修飾,否則會報錯。
  • final類:類不能被繼承,內(nèi)部的方法和變量都變成final類型

JDK和Cglib的區(qū)別:

  • jdk動態(tài)代理是利用反射機制生成一個實現(xiàn)代理接口的匿名類,在調(diào)用具體方法前調(diào)用InvokeHandler來處理
  • cglib動態(tài)代理是利用ASM開源包,對被代理對象類的class文件加載進來,通過修改其字節(jié)碼生成子類來處理
  • ASM: 一個 Java 字節(jié)碼操控框架。它能被用來動態(tài)生成類或者增強既有類的功能。ASM 可以直接產(chǎn)生二進制 class 文件,也可以在類被加載入 Java 虛擬機之前動態(tài)改變類行為。Java class 被存儲在嚴格格式定義的 .class 文件里,這些類文件擁有足夠的元數(shù)據(jù)來解析類中的所有元素:類名稱、方法、屬性以及 Java 字節(jié)碼(指令)。ASM 從類文件中讀入信息后,能夠改變類行為,分析類信息,甚至能夠根據(jù)用戶要求生成新類。 -- 以上ASM解釋來自簡書

多級動態(tài)代理

看完上面的動態(tài)代理,不知道大家有沒有想法,實現(xiàn)一個多級動態(tài)代理。

還是以螳螂捕蟬為例子,再加上一個黃雀在后,實現(xiàn)多級動態(tài)代理模式。

 
 
 
 
  1. public class Cardinal implements InvocationHandler {
  2.   // 監(jiān)聽代理代理對象
  3.     private Object proxyOne;
  4.     public Cardinal(Object proxyOne) {
  5.         this.proxyOne = proxyOne;
  6.     }
  7.     // 螳螂主要業(yè)務(wù),也就是監(jiān)聽對象
  8.     @Override
  9.     public Object invoke(Object proxy, Method method, Object[] args) throws            Throwable {
  10.         method.invoke(proxyOne, args);
  11.         secondaryMain();
  12.         return null;
  13.     }
  14.     private void secondaryMain() {
  15.         System.out.println("黃雀吃螳螂 - 次要業(yè)務(wù)");
  16.     }
  17. }

創(chuàng)建一個黃雀代理對象,那作為他的真實對象就變成螳螂了,當螳螂對象發(fā)生調(diào)用時,黃雀就能堅挺到,同時作出對應(yīng)業(yè)務(wù)邏輯

 
 
 
 
  1. public class BeanFactory {
  2.     public static BaseService newInstanc(Class classFile) {
  3.         // 1. 創(chuàng)建蟬,真實類對象
  4.         BaseService trueCicada = new Cicada();
  5.         // 2.創(chuàng)建代理類 螳螂
  6.         InvocationHandler prayingMantis = new PrayingMantis(trueCicada);
  7.         // 3.向Jvm索要代理對象 其實就是堅挺的對象
  8.         Class classArray[] = {BaseService.class};
  9.         BaseService baseService = (BaseService) Proxy.newProxyInstance(classFile.getClassLoader(), classArray, prayingMantis);
  10.         // 4.創(chuàng)建代理實現(xiàn)類 黃雀 二級代理
  11.         InvocationHandler cardinal = new Cardinal(baseService);
  12.         BaseService secondBaseService = (BaseService) Proxy.newProxyInstance(classFile.getClassLoader(), classArray, cardinal);
  13.         
  14.        // 假設(shè)要實現(xiàn)三級,四級代理,則在黃雀類上再加一層代理即可實現(xiàn)。
  15.        // 省略其他的更多級代理對象
  16.         return secondBaseService;
  17.     }
  18.   
  19.      // 測試demo
  20.       public static void main(String[] args) {
  21.         BaseService baseService  = BeanFactory.newInstanc(Cicada.class);
  22.         baseService.mainService();
  23.         // 結(jié)果:主要業(yè)務(wù)
  24.         //      螳螂捕蟬 - 次要業(yè)務(wù)
  25.         //      黃雀吃螳螂 - 次要業(yè)務(wù)
  26.     }
  27. }

根據(jù)這個代碼基本就實現(xiàn)多級代理過程。螳螂監(jiān)聽著蟬類的動作,黃雀監(jiān)聽著螳螂類的動作。

同樣的如果要實現(xiàn)三級代理,四級代理也就不是什么難事了,在每一層的上面再加一個代理對象就可以了。

  • 動態(tài)代理本質(zhì)還是可以理解為將“次要業(yè)務(wù)”與“主要業(yè)務(wù)”解耦合,讓開發(fā)者能更加專注于主要業(yè)務(wù),提升開發(fā)效率,以及維護成本。

總結(jié)

代理模式在業(yè)務(wù)代碼上我個人認為是比較少見的,特別是動態(tài)代理基本上是沒有見過。但是代理模式也是我們必須要理解的一種模式,因為學習好代理模式有助于我們?nèi)プx一些源碼,排查一些更深層次的問題,或者面對一些業(yè)務(wù)場景問題,也能有一個很大的提升,設(shè)計模式本身也就是為了解決問題而創(chuàng)建出來的。

理解完動態(tài)代理現(xiàn)在對我們來說AOP的實現(xiàn)原理也就不言而喻了。

詳細的設(shè)計模式到這里就結(jié)束了,后面針對一些不常見設(shè)計模式我還是會給大家做一個總結(jié)吧。

我是敖丙,你知道的越多,你不知道的越多,我們下期見!!!


當前題目:詳解設(shè)計模式之代理模式
轉(zhuǎn)載源于:http://www.dlmjj.cn/article/dpiidgg.html