dotnetcore實現Aop

来源:https://www.cnblogs.com/yyfh/archive/2019/09/10/11496033.html
-Advertisement-
Play Games

Asp.NetCore實現Aop,AspectCore實現Aop ...


dotnetcore實現Aop

   Aop大家都不陌生,然而今天給大家不將講官方的filter,今天給大家分享一個輕量級的Aop解決方案(AspectCore)

什麼是AspectCore

AspectCore是一個面向切麵編程,基於.NetCore和.NetFramwork的擴平臺框架,對方法攔截器、依賴項註入集成、web應用程式、數據驗證等提供核心支持。

AspectCore基本特性

  • 提供抽象的Aop介面,基於該介面可以輕鬆的使用自己的代理類實現替換預設的實現.

  • 框架不包含IoC,也不依賴具體IoC實現,可以使用Asp.Net Core的內置依賴註入或者任何相容Asp.Net Core的第三方Ioc來繼承AspectCore到Asp.NetCore應用中

  • 高性能的非同步攔截系統

  • 靈活的配置系統

  • 基於service的而非基於實現類的切麵構造

  • 支持擴平臺的Asp.Net Core環境

使用AspectCore

從NuGet中安裝AspectCore

AspectCore.Extensions.DependencyInjection

package

PM> Install-package AspectCore.Extensions.DependencyInjection

下麵我創建了一個Api應用程式.

NuGet安裝

AspectCore.Configuration

package

PM> Install-package AspectCore.Configuration

下麵我新建了一個攔截器 CustomInterceptorAttribute,繼承AbstractInterceptorAttribute(一般情況下繼承他即可),他實現IInterceptor介面AspectCore預設實現了基於Attribute的攔截器配置。

/// <summary>
///     自定義攔截器
/// </summary>
public class CustomInterceptorAttribute : AbstractInterceptorAttribute
{
    /// <summary>
    ///     實現抽象方法
    /// </summary>
    /// <param name="context"></param>
    /// <param name="next"></param>
    public override async Task Invoke(AspectContext context, AspectDelegate next)
    {
        try
        {
            Console.WriteLine("執行之前");
            await next(context);//執行被攔截的方法
        }
        catch (Exception)
        {
            Console.WriteLine("被攔截的方法出現異常");
            throw;
        }
        finally
        {
            Console.WriteLine("執行之後");
        }
    }
}

定義ICustomService介面和它的實現類CustomService:

public interface ICustomService
{
    DateTime GetDateTime();
}
public class CustomService : ICustomService
{
    public DateTime GetDateTime()
    {
        return DateTime.Now;

     }
}

在ValuesController註入ICustomService

[Route("api/[controller]")]
[ApiController]
public class ValuesController : ControllerBase
{
    private readonly ICustomService _icustomserveice;
    public ValuesController(ICustomService icustomService) {
        this._icustomserveice = icustomService;
    }

    // GET api/values
    [HttpGet]
    public DateTime Get()
    {
        return _icustomserveice.GetDateTime();
    }

}

註冊ICustomService,並創建代理容器

 public IServiceProvider ConfigureServices(IServiceCollection services)
        {
            services.AddTransient<ICustomService,CustomService>();
            services.AddMvc();
            //全局攔截器。使用AddDynamicProxy(Action<IAspectConfiguration>)的重載方法,其中IAspectConfiguration提供Interceptors註冊全局攔截器:
            services.ConfigureDynamicProxy(config=> {
                config.Interceptors.AddTyped<CustomInterceptorAttribute>();
            });
            services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);
            return services.BuildAspectInjectorProvider();
     }

作為服務的全局攔截器。在ConfigureServices中添加:

services.AddTransient<CustomInterceptorAttribute>(provider => new CustomInterceptorAttribute());

作用於特定ServiceMethod的全局攔截器,下麵的代碼演示了作用於帶有Service尾碼的類的全局攔截器:

 services.ConfigureDynamicProxy(config =>
            {
                config.Interceptors.AddTyped<CustomInterceptorAttribute>(method => method.DeclaringType.Name.EndsWith("Service"));
            });

通配符攔截器,匹配尾碼為Service

 services.ConfigureDynamicProxy(config =>
            {
                config.Interceptors.AddTyped<CustomInterceptorAttribute>(Predicates.ForService("*Service"));
            });

在AspectCore中提供NonAspectAttribute來使得ServiceMethod不被代理:

   [NonAspect]
    DateTime GetDate();

全局配置忽略條件

      services.ConfigureDynamicProxy(config =>
        {
            //Namespace命名空間下的Service不會被代理
            config.NonAspectPredicates.AddNamespace("Namespace");
            //最後一級為Namespace的命名空間下的Service不會被代理
            config.NonAspectPredicates.AddNamespace("*.Namespace");
            //ICustomService介面不會被代理
            config.NonAspectPredicates.AddService("ICustomService");
            //尾碼為Service的介面和類不會被代理
            config.NonAspectPredicates.AddService("*Service");
            //命名為Method的方法不會被代理
            config.NonAspectPredicates.AddMethod("Method");
            //尾碼為Method的方法不會被代理
            config.NonAspectPredicates.AddMethod("*Method");
        });

AspectCore: [https://github.com/dotnetcore/AspectCore-Framework]
測試項目地址: [https://github.com/fhcodegit/DotNetAspectCore/tree/master]


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

-Advertisement-
Play Games
更多相關文章
  • 前提 入行已經7,8年了,一直想做一套漂亮點的自定義控制項,於是就有了本系列文章。 GitHub:https://github.com/kwwwvagaa/NetWinformControl 碼雲:https://gitee.com/kwwwvagaa/net_winform_custom_contr ...
  • 背景:臨時提供一個簡單的網頁,供其他人瀏覽資料庫(Oracel、MSSQL)的某些數據,並導出Excel。 ...
  • 依賴註入主要是一種結構性的模式,註重的是類與類之間的結構,它要達到的目的就是設計原則中最少知道和合成復用的原則,減少內部依賴,履行單一職責,最終就是強解耦。依賴註入目前最好的實現就是依賴註入容器。 Unity是微軟Patterns & Practices團隊所開發的一個輕量級的,並且可擴展的依賴註入 ...
  • 場景 DevExpress的PanelControl常用進行窗體頁面的佈局。 一般是拖拽一個PannelControl,然後是再拖拽其他控制項。 如果是由代碼生成控制項並控制佈局的話,怎樣實現。 關註公眾號 霸道的程式猿 獲取編程相關電子書、教程推送與免費下載。 實現 比如說要在PanelContrl中 ...
  • 通過 abp(net core)+easyui+efcore實現倉儲管理系統——菜單-上 (十六)這篇文章,我們已經瞭解了ABP中的菜單相關的類及類的屬性與方法,接下我們通過實例來實現一個動態載入菜單的功能。動態菜單是我們在abp(net core)+easyui+efcore實現倉儲管理系統——領... ...
  • 場景 使用DevExpress的EditText控制項時,需要限制其輸入類型為數字。 正常來說是窗體上拖拽一個TextEdit,然後在設計視窗點擊小三角,選擇Change Mask 但是如果說TextEdit控制項不是拖拽上去而是由代碼生成的,那麼在代碼中怎樣設置只能輸入數字。 關註公眾號 霸道的程式猿 ...
  • 前提 入行已經7,8年了,一直想做一套漂亮點的自定義控制項,於是就有了本系列文章。 GitHub:https://github.com/kwwwvagaa/NetWinformControl 碼雲:https://gitee.com/kwwwvagaa/net_winform_custom_contr ...
  • Win10 IIS預設是.net 4.0,安裝VS2015後,IIS沒有.net 4.5,解決方法,直接在CMD命令行下執行下麵語句 ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...