C# 使用Emit實現動態AOP框架 (一)

来源:https://www.cnblogs.com/accode/archive/2019/05/21/10900711.html
-Advertisement-
Play Games

最近需要有一個完全自主的基於C#語言的Aop框架,查了一下資料實現方式主要分為:靜態織入和動態代理,靜態織入以Postshop為代表,而動態代理又分為: 1、普通反射 2、Emit反射 3、微軟提供的.Net Remoting和RealProxy (微軟官方例子https://msdn.micros ...


        最近需要有一個完全自主的基於C#語言的Aop框架,查了一下資料實現方式主要分為:靜態織入和動態代理,靜態織入以Postshop為代表,而動態代理又分為:

1、普通反射

2、Emit反射

3、微軟提供的.Net Remoting和RealProxy

(微軟官方例子https://msdn.microsoft.com/zh-cn/library/dn574804.aspx

      總體來說靜態織入速度最快,普通反射最慢,而.Net Remoting和RealProx實現起來又相對較複雜。而Emit速度居中,同時其一次生成後,將結果序列化,速度也並不慢,同時和原有類並沒有緊密耦合,通過外部配置文件可以方便的控制要進行代理的類型、方法和屬性,其缺點是被代理的方法、屬性必須為virtual類型。

一、被代理類和代理類

       被代理類,是我們正常使用的類,裡邊是原有的業務邏輯,只要在被代理方法上申明上相應的切麵特性就行了,使用起來比較簡單;如下

 1   public class AopTest
 2   {
 3         
 4         public AopTest()
 5         {
 6             Name = "小明"; Age = 10;
 7         }
 8 
 9         public AopTest(string name, int age)
10         {
11             Name = name; Age = age;
12         }
13 
14         [Log]
15         public virtual string Name { get; set; }
16 
17         [Log]
18         public virtual int Age { get; set; }
19 
20         [Log]
21         public virtual int NYearLater(int a)
22         {
23             int larter = Age + a;
24 
25             return larter;
26         }
27     }

 

     代理類是Aop框架自動生成的類,使用反編譯工具我們可以看到,它比被代理類多了切麵上下文聲明(AspectContent)和相應的切麵特性對象聲明,在被代理類的方法執行前後,相應切麵特性調用OnEntry、OnExit執行相關操作,如日誌、參數驗證、許可權驗證等等Aop功能,其中AspectContext是OnEntry、OnExit調用參數,如下:

public class AopTest_Proxy : AopTest
{
    public override string Name
    {
        get
        {
            object[] args = new object[0];
            AspectContext aspectContext = new AspectContext(this, "get_Name", args);
            LogAttribute logAttribute = new LogAttribute();
            logAttribute.OnEntry(aspectContext);
            string name = base.Name;
            aspectContext.Result = name;
            logAttribute.OnExit(aspectContext);
            return name;
        }
        set
        {
            AspectContext context = new AspectContext(this, "set_Name", new object[]
            {
                value
            });
            LogAttribute logAttribute = new LogAttribute();
            logAttribute.OnEntry(context);
            base.Name = value;
            logAttribute.OnExit(context);
        }
    }

    public override int Age
    {
        get
        {
            object[] args = new object[0];
            AspectContext aspectContext = new AspectContext(this, "get_Age", args);
            LogAttribute logAttribute = new LogAttribute();
            logAttribute.OnEntry(aspectContext);
            int age = base.Age;
            aspectContext.Result = age;
            logAttribute.OnExit(aspectContext);
            return age;
        }
        set
        {
            AspectContext context = new AspectContext(this, "set_Age", new object[]
            {
                value
            });
            LogAttribute logAttribute = new LogAttribute();
            logAttribute.OnEntry(context);
            base.Age = value;
            logAttribute.OnExit(context);
        }
    }

    public AopTest_Proxy(string name, int age) : base(name, age)
    {
    }

    public override int NYearLater(int num)
    {
        AspectContext aspectContext = new AspectContext(this, "NYearLater", new object[]
        {
            num
        });
        LogAttribute logAttribute = new LogAttribute();
        logAttribute.OnEntry(aspectContext);
        int num2 = base.NYearLater(num);
        aspectContext.Result = num2;
        logAttribute.OnExit(aspectContext);
        return num2;
    }
}

 

二、測試方法

public static void Test()
{
  try
  {
     AopTest WithPara = DynamicProxy.Create<AopTest>("lxl", 10); ;
 
     WithPara.NYearLater(10);
     Console.WriteLine("done...");
   }
   catch (Exception ex)
   {
      Console.WriteLine(ex.Message);
   }
 }

 

測試方法中:AopTest WithPara = DynamicProxy.Create<AopTest>("lxl", 10); ,生成一個代理對象,其他就正常使用就可以了。

調用測試方法執行結果如下:

Log OnEntry:set_Name(lxl)
Log OnExit: set_Name

Log OnEntry:set_Age(10)
Log OnExit: set_Age

Log OnEntry:NYearLater(10)
Log OnEntry:get_Age()
Log OnExit: get_Age Result: 10

Log OnExit: NYearLater Result: 20

done...

 

   通過結果可以看到 屬性Name、Age的Set方法,NYearLater方法,以及Age屬性的Get方法都實現了日誌記錄。下邊將分幾篇來詳細介紹DynamicProxy類的實現。歡迎大家多多指正、交流。

 


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

-Advertisement-
Play Games
更多相關文章
  • 簡述:QPropertyAnimation (動畫類,用來向QObject對象添加動畫) 該類的繼承框圖如下所示: 1.QAbstractAnimation(所有動畫的抽象基類) 該抽象類為QPropertyAnimation提供了動畫播放,暫停,停止,持續時間,迴圈周期等抽象函數. 其中常用的成員 ...
  • 樹的結構說得差不多了,現在我們來說說一種數據結構叫做哈希表(hash table),哈希表有是乾什麼用的呢?我們知道樹的操作的時間複雜度通常為O(logN),那有沒有更快的數據結構?當然有,那就是哈希表; 1.哈希表簡介 哈希表(hash table)是一種數據結構,提供很快速的插入和查找操作(有的 ...
  • 一、項目背景 YW公司是一家電池供應商,目前由於公司內部的需要,需要做一個CRM項目,需要每一個不同角色的員工登陸系統後處理自己的事情。其流程大致如下: 其項目包括三部分內容: 1、許可權分配組件(rbac組件) 2、各個表的curd功能組件(stark組件) 3、將業務與上述兩個組件進行融入 二、各 ...
  • 1. 前言 WPF有一個靈活的UI框架,用戶可以輕鬆地使用代碼控制控制項的外觀。例設我需要一個控制項在滑鼠進入的時候背景變成藍色,我可以用下麵這段代碼實現: 但一般沒人會這麼做,因為這樣做代碼和UI過於耦合,難以擴展。正確的做法應該是使用代碼告訴ControlTemplate去改變外觀,或者控制Cont ...
  • 準備工作完成後,DynamicProxy類就可以開始了。 創建代理對象 Create 創建代理對象主要分為五步: (1)、獲取被代理類型構造函數參數列表 1 /// <summary> 2 /// 創建代理類型 3 /// </summary> 4 /// <param name="srcType" ...
  • 那麼明顯開發模式是三種。即:DateBase First(資料庫優先)、Model First(模型優先)和Code First(代碼優先)。當然,如果把Code First模式的兩種具體方式獨立出來,那就是四種了。Code First(New DataBase) :在代碼中定義類和映射關係並通過m ...
  • ASP.NET的優勢 ASP.NET背後有一個完整的.NET Framework支撐 什麼是ASP.NET? ASP.NET是建立在公共語言運行庫上的編程框架,可用於在伺服器上生成功能強大的Web應用程式。與以前的Web開發模型相比,ASP.NET提供了數個重要的優點: MVC模式的優勢 mvc是一 ...
  • 實現DynamicProxy前,先介紹幾個必要的輔助類: 一、切麵上下文類AspectContext 該類是作為切麵特性類的OnEntry和OnEixt方法的參數用的,該類包含了被代理對象Sender、當前切入的方法名稱(Name)、調用方法的參數列表(Args)以及返回值(Result) 二、切麵 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...