AspectCore和MSDI 實現Name註冊以及解析對象

来源:https://www.cnblogs.com/jsnaner/archive/2023/04/23/17347050.html
-Advertisement-
Play Games

AspectCore 在註冊服務這塊比較簡單,預設是無法根據Name去註冊和解析對象,這邊做一下這塊的擴展 大致原理是根據自定義Name去生成對應的動態類型,然後使用委托或者對象的方式,進行註冊 tips:由於底層原理的原因,無法支持Type的方式進行註冊 定義好動態類型的介面,以及相關實現 1 p ...


AspectCore 在註冊服務這塊比較簡單,預設是無法根據Name去註冊和解析對象,這邊做一下這塊的擴展

大致原理是根據自定義Name去生成對應的動態類型,然後使用委托或者對象的方式,進行註冊

tips:由於底層原理的原因,無法支持Type的方式進行註冊

 

定義好動態類型的介面,以及相關實現

 1 public interface INamedService : IDisposable
 2 {
 3     object Service { get; }
 4 }
 5 
 6 public interface INamedService<out T> : INamedService
 7     where T : class
 8 {
 9     new T Service { get; }
10 }
11 
12 public interface INamedService<out TService, TNamed> : INamedService<TService>
13     where TService : class
14     where TNamed : struct
15 {
16 }
17 
18 internal class NamedService<TService, TNamed> : INamedService<TService, TNamed>
19     where TService : class
20     where TNamed : struct
21 {
22     ~NamedService()
23     {
24         Dispose(false);
25     }
26 
27     public NamedService(TService service)
28     {
29         Service = service;
30     }
31 
32     public TService Service { get; private set; }
33 
34     object INamedService.Service => Service;
35 
36     public void Dispose()
37     {
38         Dispose(true);
39         GC.SuppressFinalize(this);
40     }
41 
42     private void Dispose(bool disposing)
43     {
44         if (!disposing)
45         {
46             return;
47         }
48 
49         if (Service == null)
50         {
51             return;
52         }
53 
54         if (Service is IDisposable disposable)
55         {
56             disposable.Dispose();
57             Service = null;
58         }
59     }
60 }

 

根據自定義Name和ServiceType生成對應的動態類型

 1 public static class NamedBuilder
 2 {
 3     private const string KeyPrefix = "Ksd.NamedType.";
 4     private static readonly ModuleBuilder _moduleBuilder = AssemblyBuilder.DefineDynamicAssembly(new AssemblyName("Ksd.DynamicAssembly.NamedType"), AssemblyBuilderAccess.Run).DefineDynamicModule("NamedTypeModule");
 5     private static readonly ConcurrentDictionary<string, Type> _namedTypes = new(StringComparer.OrdinalIgnoreCase);
 6 
 7     private static Type GetNamedType(string name)
 8     {
 9         if (name.IsNull())
10         {
11             throw new ArgumentNullException(nameof(name));
12         }
13 
14         name = KeyPrefix + name;
15 
16         return _namedTypes.GetOrAdd(name, k =>
17         {
18             var tb = _moduleBuilder.DefineType(k, TypeAttributes.Public | TypeAttributes.Sealed, typeof(ValueType));
19             var type = tb.CreateTypeInfo().AsType();
20             return type;
21         });
22     }
23 
24     public static Type CreateServiceType(string name, Type serviceType)
25     {
26         var namedType = GetNamedType(name);
27         return typeof(NamedService<,>).MakeGenericType(serviceType, namedType);
28     }
29 
30     public static Type CreateServiceBaseType(string name, Type serviceType)
31     {
32         var namedType = GetNamedType(name);
33         return typeof(INamedService<,>).MakeGenericType(serviceType, namedType);
34     }
35 
36     public static Type GetNameServiceBaseType(Type namedType)
37     {
38         return namedType.GetInterfaces()[0];
39     }
40 }

 

封裝根據自定義的Name註冊和解析的相關方法

  1 public static class NamedServiceExtensions
  2 {
  3     #region AddInstance
  4     public static IServiceContext AddInstance(this IServiceContext services, Type serviceType, object implementationInstance, string name)
  5     {
  6         if (name.IsNull())
  7         {
  8             return services.AddInstance(serviceType, implementationInstance);
  9         }
 10 
 11         if (services == null)
 12         {
 13             throw new ArgumentNullException(nameof(services));
 14         }
 15 
 16         if (serviceType == null)
 17         {
 18             throw new ArgumentNullException(nameof(serviceType));
 19         }
 20 
 21         if (implementationInstance == null)
 22         {
 23             throw new ArgumentNullException(nameof(implementationInstance));
 24         }
 25 
 26         var namedType = NamedBuilder.CreateServiceType(name, serviceType);
 27         services.AddInstance(NamedBuilder.GetNameServiceBaseType(namedType), Activator.CreateInstance(namedType, implementationInstance));
 28         return services;
 29     }
 30 
 31     public static IServiceContext AddInstance<TService>(this IServiceContext services, TService implementationInstance, string name)
 32     {
 33         return services.AddInstance(typeof(TService), implementationInstance, name);
 34     }
 35     #endregion
 36 
 37     #region AddDelegate
 38     public static IServiceContext AddDelegate(this IServiceContext services, Type serviceType, Func<IServiceResolver, object> implementationDelegate, string name, Lifetime lifetime = Lifetime.Transient)
 39     {
 40         if (name.IsNull())
 41         {
 42             return services.AddDelegate(serviceType, implementationDelegate, lifetime);
 43         }
 44 
 45         if (services == null)
 46         {
 47             throw new ArgumentNullException(nameof(services));
 48         }
 49 
 50         if (serviceType == null)
 51         {
 52             throw new ArgumentNullException(nameof(serviceType));
 53         }
 54 
 55         if (implementationDelegate == null)
 56         {
 57             throw new ArgumentNullException(nameof(implementationDelegate));
 58         }
 59 
 60         var namedType = NamedBuilder.CreateServiceType(name, serviceType);
 61         services.AddDelegate(NamedBuilder.GetNameServiceBaseType(namedType), s => Activator.CreateInstance(namedType, implementationDelegate(s)), lifetime);
 62         return services;
 63     }
 64 
 65     public static IServiceContext AddDelegate<TService>(this IServiceContext services, Func<IServiceResolver, TService> implementationDelegate, string name, Lifetime lifetime = Lifetime.Transient)
 66         where TService : class
 67     {
 68         return services.AddDelegate(typeof(TService), implementationDelegate, name, lifetime);
 69     }
 70     #endregion
 71 
 72     #region Resolve
 73     public static T Resolve<T>(this IServiceResolver serviceResolver, string name)
 74         where T : class
 75     {
 76         if (name.IsNull())
 77         {
 78             return serviceResolver.Resolve<T>();
 79         }
 80 
 81         var namedType = NamedBuilder.CreateServiceBaseType(name, typeof(T));
 82         var namedService = serviceResolver.Resolve(namedType) as INamedService<T>;
 83         return namedService?.Service;
 84     }
 85 
 86     public static object Resolve(this IServiceResolver serviceResolver, Type serviceType, string name)
 87     {
 88         if (name.IsNull())
 89         {
 90             return serviceResolver.Resolve(serviceType);
 91         }
 92 
 93         var namedType = NamedBuilder.CreateServiceBaseType(name, serviceType);
 94         var namedService = serviceResolver.Resolve(namedType) as INamedService;
 95         return namedService?.Service;
 96     }
 97 
 98     public static IEnumerable<T> ResolveMany<T>(this IServiceResolver serviceResolver, string name)
 99         where T : class
100     {
101         if (name.IsNull())
102         {
103             return serviceResolver.ResolveMany<T>();
104         }
105 
106         var namedType = NamedBuilder.CreateServiceBaseType(name, typeof(T));
107         var namedServices = serviceResolver.ResolveMany(namedType).OfType<INamedService<T>>();
108         return namedServices.Select(t => t.Service);
109     }
110 
111     public static IEnumerable<object> ResolveMany(this IServiceResolver serviceResolver, Type serviceType, string name)
112     {
113         if (name.IsNull())
114         {
115             return serviceResolver.ResolveMany(serviceType);
116         }
117 
118         var namedType = NamedBuilder.CreateServiceBaseType(name, serviceType);
119         var namedServices = serviceResolver.ResolveMany(namedType) as IEnumerable<INamedService>;
120         return namedServices.Select(t => t.Service);
121     }
122     #endregion
123 
124     #region Remove
125     public static IServiceContext RemoveAll(this IServiceContext services, Type serviceType, string name)
126     {
127         if (name.IsNull())
128         {
129             return services.RemoveAll(serviceType);
130         }
131 
132         var namedType = NamedBuilder.CreateServiceBaseType(name, serviceType);
133         services.RemoveAll(namedType);
134         return services;
135     }
136 
137     public static IServiceContext RemoveAll<T>(this IServiceContext services, string name)
138         where T : class
139     {
140         return services.RemoveAll(typeof(T), name);
141     }
142     #endregion
143 
144     #region Other
145     public static bool Contains(this IServiceContext services, Type serviceType, string name)
146     {
147         if (name.IsNull())
148         {
149             return services.Contains(serviceType);
150         }
151 
152         var namedType = NamedBuilder.CreateServiceBaseType(name, serviceType);
153         return services.Contains(namedType);
154     }
155     #endregion
156 }

 

根據以上原理,MSDI亦可作對應的註冊以及解析方法

  1 public static class NamedServiceExtensions
  2 {
  3     #region Add
  4     public static IServiceCollection AddSingleton(this IServiceCollection services, Type serviceType, object implementationInstance, string name)
  5     {
  6         services.AddNamed(serviceType, _ => implementationInstance, name, ServiceLifetime.Singleton);
  7         return services;
  8     }
  9 
 10     public static IServiceCollection AddSingleton<TService>(this IServiceCollection services, TService implementationInstance, string name)
 11         where TService : class
 12     {
 13         return services.AddSingleton(typeof(TService), implementationInstance, name);
 14     }
 15 
 16     public static IServiceCollection AddSingleton(this IServiceCollection services, Type serviceType, Func<IServiceProvider, object> implementationFactory, string name)
 17     {
 18         services.AddNamed(serviceType, implementationFactory, name, ServiceLifetime.Singleton);
 19         return services;
 20     }
 21 
 22     public static IServiceCollection AddSingleton<TService>(this IServiceCollection services, Func<IServiceProvider, TService> implementationFactory, string name)
 23         where TService : class
 24     {
 25         return services.AddSingleton(typeof(TService), implementationFactory, name);
 26     }
 27 
 28     public static IServiceCollection AddScoped(this IServiceCollection services, Type serviceType, Func<IServiceProvider, object> implementationFactory, string name)
 29     {
 30         services.AddNamed(serviceType, implementationFactory, name, ServiceLifetime.Scoped);
 31         return services;
 32     }
 33 
 34     public static IServiceCollection AddScoped<TService>(this IServiceCollection services, Func<IServiceProvider, TService> implementationFactory, string name)
 35         where TService : class
 36     {
 37         return services.AddScoped(typeof(TService), implementationFactory, name);
 38     }
 39 
 40     public static IServiceCollection AddTransient(this IServiceCollection services, Type serviceType, Func<IServiceProvider, object> implementationFactory, string name)
 41     {
 42         services.AddNamed(serviceType, implementationFactory, name, ServiceLifetime.Transient);
 43         return services;
 44     }
 45 
 46     public static IServiceCollection AddTransient<TService>(this IServiceCollection services, Func<IServiceProvider, TService> implementationFactory, string name)
 47         where TService : class
 48     {
 49         return services.AddTransient(typeof(TService), implementationFactory, name);
 50     }
 51 
 52     public static IServiceCollection AddNamed(this IServiceCollection services, Type serviceType, Func<IServiceProvider, object> implementationFactory, string name, ServiceLifetime lifetime = ServiceLifetime.Transient)
 53     {
 54         if (name.IsNull())
 55         {
 56             services.Add(ServiceDescriptor.Describe(serviceType, implementationFactory, lifetime));
 57             return services;
 58         }
 59 
 60         if (services == null)
 61         {
 62             throw new ArgumentNullException(nameof(services));
 63         }
 64 
 65         if (serviceType == null)
 66         {
 67             throw new ArgumentNullException(nameof(serviceType));
 68         }
 69 
 70         if (implementationFactory == null)
 71         {
 72             throw new ArgumentNullException(nameof(implementationFactory));
 73         }
 74 
 75         var namedType = NamedBuilder.CreateServiceType(name, serviceType);
 76 
 77         services.Add(ServiceDescriptor.Describe(namedType, s => Activator.CreateInstance(namedType, implementationFactory(s)), lifetime));
 78         return services;
 79     }
 80     #endregion
 81 
 82     #region GetService
 83     public static T GetService<T>(this IServiceProvider serviceProvider, string name)
 84         where T : class
 85     {
 86         if (serviceProvider == null)
 87         {
 88             throw new ArgumentNullException(nameof(serviceProvider));
 89         }
 90 
 91         if (name.IsNull())
 92         {
 93             return serviceProvider.GetService<T>();
 94         }
 95 
 96         var namedType = NamedBuilder.CreateServiceBaseType(name, typeof(T));
 97         var namedService = serviceProvider.GetService(namedType) as INamedService<T>;
 98         return namedService?.Service;
 99     }
100 
101     public static object GetService(this IServiceProvider serviceProvider, Type serviceType, string name)
102     {
103         if (serviceProvider == null)
104         {
105             throw new ArgumentNullException(nameof(serviceProvider));
106         }
107 
108         if (name.IsNull())
109         {
110             return serviceProvider.GetService(serviceType);
111         }
112 
113         var namedType = NamedBuilder.CreateServiceBaseType(name, serviceType);
114         var namedService = serviceProvider.GetService(namedType) as INamedService;
115         return namedService?.Service;
116     }
117 
118     public static IEnumerable<T> GetServices<T>(this IServiceProvider serviceProvider, string name)
119         where T : class
120     {
121         if (serviceProvider == null)
122         {
123             throw new ArgumentNullException(nameof(serviceProvider));
124         }
125 
126         if (name.IsNull())
127         {
128             return serviceProvider.GetServices<T>();
129         }
130 
131         var namedType = NamedBuilder.CreateServiceBaseType(name, typeof(T));
132         var namedServices = serviceProvider.GetServices(namedType) as IEnumerable<INamedService<T>>;
133         return namedServices.Select(t => t.Service);
134     }
135 
136     public static IEnumerable<object> GetServices(

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

-Advertisement-
Play Games
更多相關文章
  • 原來的水文標題是“用 VS Code 搞 Qt6”,想想還是直接改為“Qt6”,反正這個用不用 VS Code 也能搞。雖然我知道大伙伴們都很討厭 CMake,但畢竟這廝幾乎成了 C++ 的玩家規範了。Qt 也算識大體,支持用 CMake 來構建程式。所以,只要你用的是能寫 C++ 的工具,理論上都 ...
  • 事實是這樣的,我有個介面,這個介面不能被篡改,於是想到了比較簡單的md5對url地址參數進行加密,把這個密碼當成是sign,然後服務端收到請求後,使用相同演算法也生成sign,兩個sign相同就正常沒有被篡改過。 問題的出現 介面中的參數包括userId,extUserId,時間,其中extUserI ...
  • 本篇文章主要介紹了,如何通過Nginx配置跨域,並覆蓋後端服務跨域配置。 先看下後端代碼跨域配置: 主要的目標是:不修改後端跨域配置代碼,來實現Nginx跨域指定功能變數名稱。 @Bean public CorsFilter corsFilter() { final UrlBasedCorsConfigur ...
  • 高質量編程簡介及編碼規範 高質量: 各種邊界條件考慮完備 異常情況處理,穩定性 易讀易維護 編程原則 簡單性 可讀性 生產力 編碼規範 公共符號始終要註釋 例外:實現介面的方法不需要註釋 格式化 使用gofmt(官方工具)自動格式化 註釋 代碼作用(適合公共符號) 代碼如何實現 (適合註釋實現過程) ...
  • 異常處理 panic 異常處理 ​ 如果出現了panic異常,那麼會停止當前函數的運行,然後會找recover()方法,如果沒有的話,就會報錯退出程式,如果有就會執行recover的方法體中的方法 ​ 我們可以使用defer延遲處理函數來捕獲panic異常,用recover()來從錯誤場景中恢復,必 ...
  • NGINX 備忘清單 Nginx (engine x) 是一個高性能的HTTP和反向代理web伺服器 ,同時也提供了IMAP/POP3/SMTP服務。Nginx是由伊戈爾·賽索耶夫為俄羅斯訪問量第二的Rambler.ru站點(俄文:Рамблер)開發的,公開版本1.19.6發佈於2020年12月1 ...
  • npm 備忘清單 npm 是 Node.js 的包管理工具,用來安裝各種 Node.js 的擴展。 npm 是 JavaScript 的包管理工具,也是世界上最大的軟體註冊表。有超過 60 萬個 JavaScript 代碼包可供下載,每周下載約 30 億次。npm 讓 JavaScript 開發人員 ...
  • 隨著技術的發展,ASP.NET Core MVC也推出了好長時間,經過不斷的版本更新迭代,已經越來越完善,本系列文章主要講解ASP.NET Core MVC開發B/S系統過程中所涉及到的相關內容,適用於初學者,在校畢業生,或其他想從事ASP.NET Core MVC 系統開發的人員。 經過前幾篇文章... ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...