設計模式(一) 動態代理初嘗試

来源:https://www.cnblogs.com/zhouyun-yx/archive/2019/02/26/10438383.html
-Advertisement-
Play Games

摘要 之前老是聽說動態代理,一直沒有機會好好看過,現在就把動態代理的實現邏輯和用處整理一下。首先提兩個概念,委托類和代理類。委托類就是實際業務邏輯的處理者,代理類是處於請求發起者與委托類之間的角色,所有對委托類的請求都會經過代理類。就是委托類將請求處理委托給代理類,代理類可以起到方法攔截、功能增強的 ...


  • 摘要
      之前老是聽說動態代理,一直沒有機會好好看過,現在就把動態代理的實現邏輯和用處整理一下。
    首先提兩個概念,委托類和代理類。委托類就是實際業務邏輯的處理者,代理類是處於請求發起者與委托類之間的角色,所有對委托類的請求都會經過代理類。
    就是委托類將請求處理委托給代理類,代理類可以起到方法攔截、功能增強的作用。
    實現動態代理的方式有很多,現在主流的主要jdk和cglib這兩個。下麵就用示例代碼說明下動態代理的過程,以及用動態代理實現攔截器。
  • 用JDK實現動態代理
      jdk實現動態代理的包是java.lang.reflect.*,jdk實現動態代理有限制,委托類必須要實現介面,所以先要創建一個介面
    public interface MyInterface {
    
        //jdk
        public void tes1();
    
        //cglib
        public void test2(String val);
    
    }


    創建介面實現類

    public class MyTarget implements MyInterface {
    
        //jdk
        @Override
        public void tes1() {
            System.out.println("委托類業務處理!");
        }
    
        //cglib
        @Override
        public void test2(String val) {
            System.out.println("業務處理結果:"+val);
        }
    }


    動態代理邏輯代碼

    package proxy;
    
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    
    public class MyProxyTest implements InvocationHandler {  //繼承InvocationHandler,攔截方法
    
        private Object target=null;
    
        //構造函數
        public MyProxyTest(Object target){
            this.target = target ;
        }
    
        //獲取代理類的方法
        public static Object getProxy(Object target){
            //通過proxy創建代理類,需要三個參數,委托類的類載入器、委托類實現的介面、代理類
            /**
             * 這是jdk的動態代理模式,必須要有介面
             * proxy會把代理類掛載到所有介面下麵
             * 如果委托類沒有實現任何介面會有問題,改用CGLIB的enhancer增強類做動態代理
             */
            return Proxy.newProxyInstance(target.getClass().getClassLoader(),
                    target.getClass().getInterfaces(),new MyProxyTest(target));
        }
    
        //攔截方法
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            System.out.println("進入代理邏輯!");
            System.out.println("代理類處理邏輯!");
            Object result=null;
            result = method.invoke(target,args);
            System.out.println("委托類處理完後的邏輯!");
            return result;
        }
    }

      getProxy方法就是綁定委托類和代理類之間關係的方法,這裡綁定的代理類是this當前對象,也就是MyProxyTest 的實例對象,由於實現了InvocationHandler介面,所以在方法調用之前都會
    通過invoke方法,在這裡就可以寫功能增強的代碼。method.invoke(target,args)是通過反射運行委托類的方法。

    測試用例

    public class JunitTest {
        public static void main(String[] args) {
            /**
             * jdk動態代理
             */
            System.out.println("---------jdk動態代理--------");
            //委托類
            MyTarget target = new MyTarget();
            //給委托類綁定代理類
            MyInterface proxy = (MyInterface)MyProxyTest.getProxy(target);
            proxy.tes1();
        }
    }


    運行結果

    ---------jdk動態代理--------
    進入代理邏輯!
    代理類處理邏輯!
    委托類業務處理!
    委托類處理完後的邏輯!

    我們發現進入了代理類的處理邏輯。



  • CGLIB實現動態代理
      CGLIB實現動態代理和JDK大同小異,不過CGLIB是通過創建增強類Enhancer,並且設置Enhancer的委托類和代理類來實現動態代理。CGLIB的委托類不需要實現介面。
    動態代理邏輯
    package proxy;
    
    import org.springframework.cglib.proxy.Enhancer;
    import org.springframework.cglib.proxy.MethodInterceptor;
    import org.springframework.cglib.proxy.MethodProxy;
    
    import java.lang.reflect.Method;
    
    public class MyCgLibProxy implements MethodInterceptor {
    
        public Object getProxy(Class target){
            //創建增強類
            Enhancer enhancer = new Enhancer();
            //設置委托類
            enhancer.setSuperclass(target);
            //設置代理類
            enhancer.setCallback(this);
            return enhancer.create();
        }
    
    
        @Override
        public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
            System.out.println("進入代理邏輯!");
            System.out.println("代理類處理邏輯!");
            Object result=null;
            result = methodProxy.invokeSuper(o,objects);
            System.out.println("委托類處理完後的邏輯!");
            return result;
        }
    }

    這裡的getProxy接收的不是委托類的實例對象而是委托類。代理類是this,MyCgLibProxy 的實例對象,由於實現了MethodInterceptor攔截器,所以方法調用都會經過intercept。

    測試用例

    package proxy;
    
    public class JunitTest {
        public static void main(String[] args) {
            /**
             * cglib動態代理
             */
            System.out.println("---------cglib動態代理--------");
            MyCgLibProxy cgLib = new MyCgLibProxy();
            MyInterface cgLibProxy = (MyInterface)cgLib.getProxy(MyTarget.class);
            cgLibProxy.test2("cglib動態代理");
        }
    }

    運行結果

    ---------cglib動態代理--------
    進入代理邏輯!
    代理類處理邏輯!
    業務處理結果:cglib動態代理
    委托類處理完後的邏輯!

    簡單的實現動態代理的兩種方式就完成了,之後再講解攔截器和責任鏈模式

  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 實現思路: 1、 v model 一個收集所有input(除全選框外)數組checkModel ,vue會動態將其checked為true的input的value值存入數組checkModel里 2 、watch函數來監聽checkModel 屬性,當其長度==input元素時 全選按鈕選中 否則取 ...
  • 1. 在 router目錄下 的 index.js文件中,新增路由 import City from '@/pages/city/City' { path: '/city', name: 'City', component: City } 2. 在city 目錄下新建city文件夾,然後新建 Cit ...
  • ├── README.md 項目介紹├── index.html 入口頁面├── build 構建腳本目錄│ ├── webpack.base.conf.js webpack基礎配置,開發環境,生產環境都依賴 │ ├── webpack.dev.conf.js webpack開發環境配置│ ├── ...
  • 前面介紹了Util是如何封裝以降低Angular應用的開發成本。 現在把關註點移到服務端,本文將介紹分層架構各構造塊及基類,並對不同層次的開發人員應如何進行業務開發提供一些建議。 Util分層架構介紹 為了控制業務邏輯複雜性,Util引入了DDD分層架構,這意味著如果你想使用DDD,Util會為你提 ...
  • 需求緣起(用一個公司的發展作為背景) 1.還是個小公司的時候,註冊用戶就20w,每天活躍用戶1w,每天最大單表數據量就1000,然後高峰期每秒併發請求最多就10,此時一個16核32G的伺服器,每秒請求支撐在2000左右,負載合理,沒有太大壓力,基本沒有宕機風險。 2.當註冊用戶達到2000W,每天活 ...
  • 一.Django的視圖函數view 一個視圖函數(類),簡稱視圖,是一個簡單的Python函數(類),它接受WEB請求並返回Web響應. 響應可以是一張網頁的HTML內容,一個重定向,一個404錯誤,一個XML文檔,或者一張圖片. 無論視圖本身包含什麼邏輯,都要返迴響應.代碼寫在哪裡也無所謂,只要它 ...
  • 本篇文章主要分析,架構師在設計系統架構時,應關心哪些關鍵要素? 一 業務場景 A公司是一家服裝公司,主要提供服裝一體化服務(服裝設計,服裝銷售,售後服務等),該公司主要通過淘寶,天貓,京東等平臺進行銷售,由於公司 良好的服裝質量,高效的服務水平和良好的信譽等,使得公司的銷售量不斷地增長,到了第五年, ...
  • 1.什麼是命名空間,官方文檔定義為: 什麼是命名空間?從廣義上來說,命名空間是一種封裝事物的方法。在很多地方都可以見到這種抽象概念。例如,在操作系統中目錄用來將相關文件分組,對於目錄中的文件來說,它就扮演了命名空間的角色。具體舉個例子,文件 foo.txt 可以同時在目錄/home/greg 和 / ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...