問題
如何使用 ASP.NET Core 服務容器進行依賴注入?
答案
建立一個服務
public interface IGreetingService { string Greet(string to); } public class GreetingService : IGreetingService { public string Greet(string to) { return $"Hello {to}"; } }
然後可以在需要的時候注入,下面將此服務注入一箇中介軟體(Middleware):
public class HelloWorldMiddleware { private readonly RequestDelegate _next; public HelloWorldMiddleware(RequestDelegate next) { _next = next; } public async Task Invoke(HttpContext context, IGreetingService greetingService) { var message = greetingService.Greet("World (via DI)"); await context.Response.WriteAsync(message); } }
使用此中介軟體的擴充套件方法(IApplicationBuilder):
public static class UseMiddlewareExtensions { public static IApplicationBuilder UseHelloWorld(this IApplicationBuilder app) { return app.UseMiddleware<HelloWorldMiddleware>(); } }
下面需要將此服務新增到ASP.NET Core的服務容器中,位於Startup.cs檔案的ConfigureServices()方法:
public void ConfigureServices(IServiceCollection services) { services.AddScoped<IGreetingService, GreetingService>(); }
然後在請求管道中(request pipeline)使用此中介軟體,位於Startup.cs檔案的Configure()方法:
public void Configure(IApplicationBuilder app, IHostingEnvironment env) { app.UseHelloWorld(); }
執行,此時頁面輸出:
建立一個帶輸入引數的服務
如果你的服務需要更復雜的初始化引數,下面我們建立一個FlexibleGreetingService:
public class FlexibleGreetingService : IGreetingService { private readonly string _sayWhat; public FlexibleGreetingService(string sayWhat) { _sayWhat = sayWhat; } public string Greet(string to) { return $"{_sayWhat} {to}"; } }
我們可以使用AddScoped的一個過載工廠方法來新增此服務到容器中:
public void ConfigureServices(IServiceCollection services) { services.AddScoped<IGreetingService, FlexibleGreetingService>(factory => { return new FlexibleGreetingService("Hi"); }); }
執行,此時頁面輸出:
如果是單件生命週期,還有一個接受服務例項的過載方法:
public void ConfigureServices(IServiceCollection services) { services.AddSingleton<IGreetingService>(new FlexibleGreetingService("Hi ")); }
討論
ASP.NET Core內建了一個輕量級的服務容器。我們可以在Startup.cs類的ConfigureServices()方法中配置需要的服務。這個方法在Configure()方法之前執行,所以我們可以在任意中介軟體使用之前配置的服務(包含MVC服務)。
依賴注入預設是通過公開建構函式來完成的,大多數情況下這是最佳實踐。
服務的生命週期
服務容器管理著新增到伺服器列表的生命週期。下面列出了新增服務的三種方法:
- AddScoped():服務會在一個請求內部只建立一次。
- AddTransient():服務會在每次需要時建立一次。
- AddSingleton():服務會在第一次需要時建立一次,並在隨後保持不變。
注:EF的生命週期應該是Scoped,我們可以通過IServiceCollection.AddDbContext來建立EF服務(內部也是作為Scoped實現)。
工廠方法
上面的方法都有一個過載方法來使用工廠方法來新增服務。對於需要複雜配置的服務這是很有用的。
這些方法的簽名看起來如下所示:
AddScoped(Func<IServiceProvider, TService>)
框架提供的服務
ConfigureServices()接受的IServiceCollection引數擁有很多內建的服務(由框架提供),可以參考ASP.NET Core文件。
IServiceCollection有很多有用的擴充套件方法來新增常用服務,比如AddDbContext,AddIdentity,AddOptions和AddMvc。
銷燬服務
服務容器會自動呼叫所有實現了IDisposable介面的服務型別,除了那些作為例項(而不是型別)新增的服務。
獲取服務(Request Services)
儘管通過建構函式來注入服務被認為是最佳實踐,我們依然可以通過IServiceProvider的GetService方法來獲取服務。在中介軟體中IServiceProvider物件可以通過HttpContext來獲取:
public async Task Invoke(HttpContext context) { var greetingService = context.RequestServices.GetService<IGreetingService>(); var message = greetingService.Greet("World (via GetService)"); await context.Response.WriteAsync(message); }
注:需要新增Microsoft.Extensions.DependencyInjection引用才能上述使用GetService的泛型過載方法。
執行,此時頁面輸出:
原始碼下載
原文:https://tahirnaushad.com/2017/08/15/asp-net-core-dependency-injection/