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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...