Metalama簡介2.利用Aspect在編譯時進行消除重覆代碼

来源:https://www.cnblogs.com/chsword/archive/2022/04/12/metalama_2.html
-Advertisement-
Play Games

本文對Metalama中的切麵進行簡介及以WPF中的 INotifyPropertyChanged 為例,展示如何利用Metalama簡化INotifyPropertyChanged 的實現 ...


上文介紹到AspectMetalama的核心概念,它本質上是一個編譯時的AOP切片。下麵我們就來系統說明一下Metalama中的Aspect
Metalama簡介1. 不止是一個.NET跨平臺的編譯時AOP框架

本文講些什麼

  1. 關於Metalama中Aspect的基礎
  2. 一些關於Aspect的示例,最終目的是通過本篇的介紹,將在編譯時自動為類型添加INotifyPropertyChanged,實現如下效果:
    1. 自動添加介面
    2. 自動添加介面實現
    3. 改寫屬性的set和get

image

關於Aspect

在前面的文章中我們已經介紹了使用Metalama編寫簡單的AOP。但是例子過於簡單,也只是在代碼前後加了兩個Console.WriteLine,並沒有太大的實際參考意義。下麵我就以幾個實際例子,來體現Metalama在復用代碼方面的好處。
對於Metalama中的Aspect分為以下兩種API

1.Aspect基礎API

  • TypeAspect 對類型進行編譯時代碼插入,見示例3
  • MethodAspect
  • PropertyAspect
  • ParameterAspect
  • EventAspect
  • FieldAspect
  • FieldOrPropertyAspect
  • ConstructorAspect

2.Override API(重寫式API)

重寫試API使用更方便、更直觀,與上面基礎API等價,但是更容易使用

  • OverrideMethodAspect 對方法進行編譯時代碼插入,請見下麵示例1
  • OverrideFieldOrPropertyAspect 對欄位或屬性進行編譯時代碼插入,請見下麵示例2
  • OverrideEventAspect 對事件進行編譯時插入代碼

MethodAspectOverrideMethodAspect 為例,以下代碼等價。

基礎API MethodAspect

    public class LogAttribute : MethodAspect
    {
        public override void BuildAspect(IAspectBuilder<IMethod> builder)
        {
           // 為方法添加重寫
           builder.Advices.OverrideMethod(builder.Target,nameof(this.MethodLog));
        }
        [Template]// 這個Template必須要加
        public dynamic MethodLog()
        {
            Console.WriteLine(meta.Target.Method.ToDisplayString() + " 開始運行.");
            var result = meta.Proceed();
            Console.WriteLine(meta.Target.Method.ToDisplayString() + " 結束運行.");
            return result;
        }
    }

Override API

    public class LogAttribute : OverrideMethodAspect
    {
        public override dynamic? OverrideMethod()
        {
            Console.WriteLine(meta.Target.Method.ToDisplayString() + " 開始運行.");
            var result = meta.Proceed();
            Console.WriteLine(meta.Target.Method.ToDisplayString() + " 結束運行.");
            return result;
        }
    }

下麵針對各種情況舉一些試例。
根據每個例子的不同也分別介紹如何對方法、欄位、屬性進行重寫。

關於meta類

通過上面的示例我們可以看到,無論是在基礎API中還是Override API中,在定義AOP方法時,都使用到了metameta是一個方便在Aspect中訪問當前AOP上下文的工具類
常用的成員有:

成員 說明
meta.Proceed() 等同於執行AOP作用目標直接執行,例如方法Aspect中就是原方法直接執行,屬性的get中就是獲取值,屬性的Set中就是賦值value
meta.Target 當前AOP的作用目標,如作用目標是個方法則通過 meta.Target.Method 調用,如果目標是個屬性則通過 meta.Target.Propery 調用
meta.This 等同於使用在AOP作用目標中的this,例如可以用於獲取AOP目標所在類的其它屬性,方法
meta.ThisStatic 用於訪問AOP作用目標中的靜態類型

示例1對方法:實現一個重試N次的功能

在平時的代碼中,有這種場景,例如,我調用一個方法或API,他有一定的概率失敗,例如發生了網路異常,所以我們就要設定一個重試機制(以重試3次然後放棄為例)。
假設我們有一個方法,代碼詳見示例中的RetryDemo

    static int _callCount;
    // 此方法第一二次調用會失敗,第三次會成功
    static void MyMethod()
    {
        _callCount++;
        Console.WriteLine($"當前是第{_callCount}次調用.");
        if (_callCount <= 2)
        {
            Console.WriteLine("前兩次直接拋異常:-(");
            throw new TimeoutException();
        }
        else
        {
            Console.WriteLine("成功 :-)");
        }
    }

如果我們直接編寫代碼,可以使用類似以下邏輯處理。

        for (int i = 0; i < 3; i++)
        {
            try
            {
                MyMethod();
                break;
            }
            catch (Exception ex)
            {
                // Console.WriteLine(ex);
            }
        }

這樣的話,對於不同的方法我們就會出現大量的重試邏輯。
那麼使用Metalama我們如何進行代碼改造,去掉復用代碼呢。
第一步,我們需要創建一個可以修改方法的AOP的Attribute,如下:

internal class RetryAttribute : OverrideMethodAspect
{
    // 重試次數
    public int RetryCount { get; set; } = 3;
    // 應用到方法的切麵模板
    public override dynamic? OverrideMethod()
    {
        for (var i = 0; ; i++)
        {
            try
            {
                return meta.Proceed(); // 這是實際調用方法的位置
            }
            catch (Exception e) when (i < this.RetryCount)
            {
                Console.WriteLine($"發生異常 {e.Message.GetType().Name}. 1秒後重試.");
                Thread.Sleep(1000);
            }
        }
    }
}

這裡可以看到定義這個Attribute時,使用了Metalama提供的基類OverrideMethodAspect此基類是用於為方法添加編譯時切麵代碼的Attribute.
然後我們將這個Attribute加到方法定義上。

    static int _callCount;

    [Retry(RetryCount = 5)]
    static void MyMethod()
    {
        _callCount++;
        Console.WriteLine($"當前是第{_callCount}次調用.");
        if (_callCount <= 2)
        {
            Console.WriteLine("前兩次直接拋異常:-(");
            throw new TimeoutException();
        }
        else
        {
            Console.WriteLine("成功 :-)");
        }
    }

這樣在編譯時Metalama就會將代碼編譯為如下圖所示。

image

RetryAttribute編譯後則會變為

image

也就是會將原有的OverrideMethod自動實現為throw new System.NotSupportedException("Compile-time-only code cannot be called at run-time.")
最終調用結果為

當前是第1次調用.
前兩次直接拋異常:-(
發生異常 String. 1秒後重試.
當前是第2次調用.
前兩次直接拋異常:-(
發生異常 String. 1秒後重試.
當前是第3次調用.
成功 :-)

源代碼:https://github.com/chsword/metalama-demo/tree/main/src/RetryDemo

示例2對屬性:INotifyPropertyChanged自動屬性的實現

在很多處理邏輯中我們會用到INotifyPropertyChanged如我們要獲取以下類的屬性更改:

public class MyModel
{
    public int Id { get; set; }
    public string Name { get; set; }
}

我們可以這麼做:

using System.ComponentModel;
public class MyModel: INotifyPropertyChanged
{
    private int _id { get; set; }
    public int Id {
        get {
            return _id;
        }
        set
        {
            if (this._id != value)
            {
                this._id = value;
                this.PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("Id"));
            }
        }
    }
    private string _name;
    public string Name
    {
        get
        {
            return _name;
        }
        set
        {
            if (this._name != value)
            {
                this._name = value;
                this.PropertyChanged?.Invoke(this, new PropertyChangedEventArgs("Name"));
            }
        }
    }
    public event PropertyChangedEventHandler? PropertyChanged;
}

但是這裡,要將自動屬性進行展開,並產生大量欄位,對於這裡的重覆代碼,我們可以用Metalama進行處理,我們最終要代碼實現為如下:

public class MyModel: INotifyPropertyChanged
{
    [NotifyPropertyChanged]
    public int Id { get; set; }
    [NotifyPropertyChanged]
    public string Name { get; set; }

    public event PropertyChangedEventHandler? PropertyChanged;
}

當然我們也要實現NotifyPropertyChangedAttribute:

public class NotifyPropertyChangedAttribute : OverrideFieldOrPropertyAspect
{
    public override dynamic OverrideProperty
    {
        // 保留原本get的邏輯
        get => meta.Proceed();
        set
        {
            // 判斷當前屬性的Value與傳入value是否相等
            if (meta.Target.Property.Value != value)
            {
                // 原本set的邏輯
                meta.Proceed();
                // 這裡的This等同於調用類的This
                meta.This.PropertyChanged?.Invoke(meta.This, new PropertyChangedEventArgs(meta.Target.Property.Name));
            }
        }
    }
}

這樣就可以實現上面相同的效果。

源代碼:https://github.com/chsword/metalama-demo/tree/main/src/PropertyDemo

示例3對類型:進一步實現INotifyPropertyChanged自動屬性

剛纔對屬性在編譯時生成INotifyPropertyChanged實現的代碼中,其實可以再進一步優化,INotifyPropertyChanged介面的實現也可以通過Metalama進一步省去,最終代碼為:

[TypeNotifyPropertyChanged]
public class MyModel
{
    public int Id { get; set; }
    public string Name { get; set; }
}

那麼TypeNotifyPropertyChangedAttribute又應該怎麼實現呢,Type Aspect並沒有對應的Override實現,所以要使用TypeAspect。

internal class TypeNotifyPropertyChangedAttribute : TypeAspect
{
    public override void BuildAspect(IAspectBuilder<INamedType> builder)
    {
        // 當前類實現一個介面
        builder.Advices.ImplementInterface(builder.Target, typeof(INotifyPropertyChanged));
        // 獲取所有符合要求的屬性
        var props = builder.Target.Properties.Where(p => !p.IsAbstract && p.Writeability == Writeability.All);
        foreach (var property in props)
        {
            //用OverridePropertySetter重寫屬性或欄位
            //參數1 要重寫的屬性 參數2 新的get實現 參數3 新的set實現
            builder.Advices.OverrideFieldOrPropertyAccessors(property, null, nameof(this.OverridePropertySetter));
        }
    }
    // Interface 要實現什麼成員
    [InterfaceMember]
    public event PropertyChangedEventHandler? PropertyChanged;

    // 也可以沒有這個方法,直接調用 meta.This 這裡只是展示另一種調用方式,更加直觀
    [Introduce(WhenExists = OverrideStrategy.Ignore)]
    protected void OnPropertyChanged(string name)
    {
        this.PropertyChanged?.Invoke(meta.This, new PropertyChangedEventArgs(name));
    }

    // 重寫set的模板
    [Template]
    private dynamic OverridePropertySetter(dynamic value)
    {
        if (value != meta.Target.Property.Value)
        {
            meta.Proceed();
            this.OnPropertyChanged(meta.Target.Property.Name);
        }

        return value;
    }
}

這樣就可以實現和以上相同效果的代碼,以後再添加實現INotifyPropertyChanged的類,只要添加以上Attribute即可。

源代碼:https://github.com/chsword/metalama-demo/tree/main/src/TypeDemo

減少代碼入侵

上面的示例3中,其實對方法還是有一定入侵的,至少要標記一個Attribute,Metalama還提供了其它無入侵的方式來為類或方法添加Aspect,我們將在後面來介紹。

先上個代碼

internal class Fabric : ProjectFabric
{
    public override void AmendProject(IProjectAmender amender)
    {
        // 添加 TypeNotifyPropertyChangedAttribute 到符合規則的類上
        // 當前篩選以 Model 結尾的本項目中的類型添加 TypeNotifyPropertyChangedAttribute
         amender.WithTargetMembers(c =>
            c.Types.Where(t => t.Name.EndsWith("Model"))
            ).AddAspect(t => new TypeNotifyPropertyChangedAttribute());
    }
}

調試

調試 Aspect 的 Attribute時,尚不能使用斷點直接調試,但可以通過以下方法:
在編譯配置中除DebugRelease外還有一個LamaDebug。選擇使用LamaDebug即可直接對Metalama的項目進行調試。

  1. 在編譯時就會調用的內容中,如BuildAspect,使用 System.Diagnostics.Debugger.Break().
  2. 在Template方法或Override中, 使用meta.DebugBreak

如果是想以附加進程等方式添加斷點調試,可以參考官方文檔https://doc.metalama.net/aspects/debugging-aspects

供大家學習參考,轉文章隨意--重典
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、序言 在Web開發中,總有一些介面需要暴露在用戶認證前訪問,簡訊發送介面特別是簡訊驗證碼註冊介面便是其中典型的一類,這類介面具有如下特點: 流量在用戶認證之前 流量在用戶認證之前,意味著無法獲取用戶ID等唯一標識符信息對流量限流 手機號未知 手機號未知意味著無法對待發送簡訊的手機號做精準檢測,判 ...
  • 記錄一些自己犯過的錯誤和一些異常信息 org.apache.ibatis.binding.BindingException: Invalid bound statement (not found): 綁定異常 mapper和xml文件綁定問題 第一步:先看看xml文件的namespace是否對應著m ...
  • 不是特別完整和齊全,自己的一些小感悟,希望能幫助大家。對新手很友好,哈哈哈。輸入:Scanner in=new Scanner(System.in); 新創建一個輸入的Scanner對象,然後賦值給in,這個作用就是獲取控制台的輸入!!! in.nextInt()表示讀入一個整數 int a; 表示 ...
  • 博客推行版本更新,成果積累制度,已經寫過的博客還會再次更新,不斷地琢磨,高質量高數量都是要追求的,工匠精神是學習必不可少的精神。因此,大家有何建議歡迎在評論區踴躍發言,你們的支持是我最大的動力,你們敢投,我就敢肝 ...
  • unity 編輯器擴展簡單入門 通過使用編輯器擴展,我們可以對一些機械的操作實現自動化,而不用使用額外的環境,將工具與開發環境融為一體;並且,編輯器擴展也提供GUI庫,來實現可視化操作;編輯器擴展甚至也可以“補充”IDE缺失的一些內容,讓IDE更加人性化。 主要內容 MenuItem無界面操作 視窗 ...
  • VBScript是Visual Basic Script的簡稱,即 Visual Basic 腳本語言,有時也被縮寫為VBS。 ...
  • 1. 前言 上一篇文章我們瞭解了bombardier,並知道了bombardier.yml與開源項目bombardier的關係,接下來的文章我們瞭解一下wrk、wrk2,並對比一下它們與bombardier的關係 2. 認識wrk wrk是一種現代 HTTP 基準測試工具,能夠在單個多核 CPU 上 ...
  • 1.舉例 為了方便加深理解,我準備先講解一個例子來映襯出CLS的主要作用。 每個國家在成立之後肯定會於他國之間建立外交關係從而進行一些貿易往來,剛開始可能只會存在與一兩個國家達成了合作,所以只會與少數的國家進行貿易往來。在進行貿易往來前為了溝通順暢,國家之間交流的語言必須要一致,這個時候通常的方式是 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...