如何在.net6webapi中使用自動依賴注入
IOC(Inversion of Control)控制反轉:控制反正是一種設計思想,旨在將程序中的控制權從程序員轉移到了容器中。容器負責管理對象之間的依賴關系,使得對象不再直接依賴于其他對象,而是通過依賴注入的方式來獲取所需的資源。
DI(Dependency Injection)依賴注入:他是IOC的具體實現方式之一,使用最為廣泛,DI通過在運行時動態地將某個依賴關系抽象為獨立的組件,提交到容器之中,需要使用時再由容器注入,提升組件重用的頻率,為系統搭建一個靈活,可擴展的平臺。
IOC/DI是一種設計模式,用于解耦組件之間的依賴關系。在傳統的編程模式中,組件之間的依賴關系是硬編碼在代碼中的,這樣會導致代碼的耦合度很高,難以維護和發展。而IOC/DI模式則是通過將組件之間的依賴關系交給容器來管理,組件不再直接依賴其他組件,而是通過容器來獲取所依賴的對象。這樣可以使組件之間的依賴關系更加靈活,容器可以根據需要動態地創建和管理組件,從而實現更好的可維護性和可擴展性。
如何在.net6webapi中使用依賴注入?首先我們定義一個服務接口及對應的實現
public interface ITestServices {int return123(); } public class TestServices : ITestServices {public int return123(){ return 123;} }然后我們在Program.cs注入服務實現
var builder = WebApplication.CreateBuilder(args);// Add services to the container.builder.Services.AddControllers();// Learn more about configuring Swagger/OpenAPI at https://aka.ms/aspnetcore/swashbucklebuilder.Services.AddEndpointsApiExplorer();builder.Services.AddSwaggerGen();builder.Services.AddTransient<ITestServices, TestServices>();var app = builder.Build();// Configure the HTTP request pipeline.if (app.Environment.IsDevelopment()){ app.UseSwagger(); app.UseSwaggerUI();}app.UseHttpsRedirection();app.UseAuthorization();app.MapControllers();app.Run();值得注意的是依賴注入有三種生命周期
作用域(Scoped):在應用程序啟動時創建,并在應用程序關閉時銷毀。這種類型的服務實例會被容器管理,但是只會被當前請求使用。當請求結束時,該服務實例會被銷毀。單例(Singleton):在應用程序啟動時創建,并在整個應用程序運行期間保持不變。這種類型的服務實例會被容器管理,并且可以被多個請求共享。瞬時(Transient):在應用程序啟動時創建,并在應用程序關閉時銷毀。這種類型的服務實例不會被容器管理,也不會被其他服務引用。最后在需要使用的控制器中構造函數注入就可以使用了
[Route('[controller]/[action]')] [ApiController] public class TestController : ControllerBase {private readonly ITestServices _testServices;public TestController(ITestServices testServices){ _testServices= testServices;}[HttpGet]public int Get123() => _testServices.return123(); }依賴注入好歸好,就是每個服務都得在Program.cs注入服務實現,一但服務多起來,麻煩不說,Program.cs中的代碼更是會變得凌亂不堪,可能會有小伙伴說,可以開一個擴展函數單獨做注入,但私以為,既然有一種方法可以一勞永逸,何樂而不為呢?
其實現便是利用.net的高級特性之一,反射
首先我們定義三個生命周期接口,其對應依賴注入的三種生命周期
//瞬時注入服務接口 public interface ITransient { } //作用域注入服務接口 public interface IScoped { } //單例注入服務接口 public interface ISingleton { }然后我們定義自動注入的擴展方法,其為核心實現
public static IServiceCollection RegisterAllServices(this IServiceCollection services){ //獲取當前程序集 var entryAssembly = Assembly.GetEntryAssembly(); //獲取所有類型 //!. null包容運算符,當你明確知道表達式的值不為null 使用!.(即null包容運算符)可以告知編譯器這是預期行為,不應發出警告 //例: entryAssembly!.GetReferencedAssemblies() 正常 //entryAssembly.GetReferencedAssemblies() 編譯器判斷entryAssembly有可能為null,變量下方出現綠色波浪線警告 var types = entryAssembly!.GetReferencedAssemblies()//獲取當前程序集所引用的外部程序集.Select(Assembly.Load)//裝載.Concat(new List<Assembly>() { entryAssembly })//與本程序集合并.SelectMany(x => x.GetTypes())//獲取所有類.Distinct();//排重 //三種生命周期分別注冊 Register<ITransient>(types, services.AddTransient, services.AddTransient); Register<IScoped>(types, services.AddScoped, services.AddScoped); Register<ISingleton>(types, services.AddSingleton, services.AddSingleton); return services;}/// <summary>/// 根據服務標記的生命周期interface,不同生命周期注冊到容器里面/// </summary>/// <typeparam name='TLifetime'>注冊的生命周期</typeparam>/// <param name='types'>集合類型</param>/// <param name='register'>委托:成對注冊</param>/// <param name='registerDirectly'>委托:直接注冊服務實現</param>private static void Register<TLifetime>(IEnumerable<Type> types, Func<Type, Type, IServiceCollection> register, Func<Type, IServiceCollection> registerDirectly){ //找到所有標記了Tlifetime生命周期接口的實現類 var tImplements = types.Where(x => x.IsClass && !x.IsAbstract && x.GetInterfaces().Any(tinterface => tinterface == typeof(TLifetime))); //遍歷,挨個以其他所有接口為key,當前實現為value注冊到容器中 foreach (var t in tImplements) {//獲取除生命周期接口外的所有其他接口var interfaces = t.GetInterfaces().Where(x => x != typeof(TLifetime));if (interfaces.Any()){ foreach (var i in interfaces) {register(i, t); }}//有時需要直接注入實現類本身registerDirectly(t); }}其核心邏輯便是通過反射掃描程序集,當掃描到實現了我們定義的生命周期接口時,為其實現對應的生命周期注入。
注冊這個服務
builder.Services.RegisterAllServices();然后我們就可以通過繼承生命周期接口來實現自動服務注入
public interface ITestServices {int return123(); } public class TestServices : ITestServices, ITransient {public int return123(){ return 123;} }接下來無需在Program.cs注入服務實現
調用成功。
到此這篇關于如何在.net6webapi中實現自動依賴注入的文章就介紹到這了,更多相關.net6webapi自動依賴注入內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!
