Dependency injection in IApplicationBuilder
Dependency injection in IEndpointRouteBuilder
Dependency injection in Middleware
ui
Dependency injection in IHtmlHelper
Dependency injection in IApplicationBuilder
1.
come from:
namespace Nop.Core.Infrastructure
{
public static class CustomServiceProvider
{
/// <summary>
/// 提供EngineContext,被Nop.Core 等用到
/// </summary>
/// <param name="services"></param>
/// <returns></returns>
public static IServiceCollection AddDI(this IServiceCollection services)
{
if (services == null)
{
throw new ArgumentNullException(nameof(services));
}
return services;
}
/// <summary>
/// see :http://www.th7.cn/Program/net/201611/1018772.shtml
/// </summary>
/// <param name="builder"></param>
/// <returns></returns>
public static IApplicationBuilder UseDI(this IApplicationBuilder builder)
{
//old
// IServiceProvider ApplicationServiceProvider= builder.ApplicationServices;
//// EngineContext.ServiceProvider = builder.ApplicationServices;
// IServiceScope scope = ApplicationServiceProvider.CreateScope();
// IServiceProvider newServiceProvider = scope.ServiceProvider;
// EngineContext.ServiceProvider = newServiceProvider;
//new
//come from:https://www.iaspnetcore.com/Blog/BlogPost/591d0d3184cd451d30d3c8e1/all-kind-of-dependency-solution-in-asp-net-core
var serviceScopeFactory = builder.ApplicationServices.GetRequiredService<IServiceScopeFactory>();
var serviceScope = serviceScopeFactory.CreateScope();
EngineContext.ServiceProvider = serviceScope.ServiceProvider;
return builder;
}
}
public static class EngineContext
{
public static IServiceProvider ServiceProvider
{
get;
set;
}
}
}
2.
/// <summary>
/// Configure the using of added middleware
/// </summary>
/// <param name="application">Builder for configuring an application's request pipeline</param>
/// <param name="webHostEnvironment">WebHostEnvironment</param>
public void Configure(IApplicationBuilder application, IWebHostEnvironment webHostEnvironment)
{
var serviceProvider = application.ApplicationServices;
var grandConfig = serviceProvider.GetRequiredService<GrandConfig>();
.Net version >= .Net core 3.x
come from:https://github.com/MiniProfiler/dotnet/blob/main/samples/Samples.AspNetCore3/Startup.cs
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
var serviceScopeFactory = app.ApplicationServices.GetRequiredService<IServiceScopeFactory>();
using (var serviceScope = serviceScopeFactory.CreateScope())
{
var dbContext = serviceScope.ServiceProvider.GetService<SampleContext>();
dbContext.Database.EnsureCreated();
}
}
Dependency injection in IEndpointRouteBuilder
.Net version>= .Net core 5.x
public static void MapReverseProxy(this IEndpointRouteBuilder endpoints, Action<IApplicationBuilder> configureApp)
{
if (endpoints is null)
{
throw new ArgumentNullException(nameof(endpoints));
}
if (configureApp is null)
{
throw new ArgumentNullException(nameof(configureApp));
}
var appBuilder = endpoints.CreateApplicationBuilder();
appBuilder.UseMiddleware<DestinationInitializerMiddleware>();
configureApp(appBuilder);
appBuilder.UseMiddleware<ProxyInvokerMiddleware>();
var app = appBuilder.Build();
var routeBuilder = endpoints.ServiceProvider.GetRequiredService<IRuntimeRouteBuilder>();
routeBuilder.SetProxyPipeline(app);
var dataSource = (EndpointDataSource)endpoints.ServiceProvider.GetRequiredService<IProxyDynamicEndpointDataSource>();
endpoints.DataSources.Add(dataSource);
}
source:github
.Net version>= .Net core 3.x
using System.Linq;
using Microsoft.AspNetCore.Routing;
using Microsoft.Extensions.DependencyInjection;
using Nop.Core.Domain.Localization;
using Nop.Data;
using Nop.Services.Localization;
namespace Nop.Web.Infrastructure
{
public class BaseRouteProvider
{
protected string GetRouterPattern(IEndpointRouteBuilder endpointRouteBuilder, string seoCode = "")
{
if (DataSettingsManager.DatabaseIsInstalled)
{
var localizationSettings = endpointRouteBuilder.ServiceProvider.GetRequiredService<LocalizationSettings>();
if (localizationSettings.SeoFriendlyUrlsForLanguagesEnabled)
{
var langservice = endpointRouteBuilder.ServiceProvider.GetRequiredService<ILanguageService>();
var languages = langservice.GetAllLanguages().ToList();
return "{language:lang=" + languages.FirstOrDefault().UniqueSeoCode + $"}}/{seoCode}";
}
}
return seoCode;
}
}
}
source:github
Dependency injection in Middleware
所有中间件的实例声明只有一Application启动时声明一次,而中间件的Invoke方法是每一次请求都会调用的。如果以Scope或者Transient方式声明依赖的对象在中间件的属性或者构造函数中注入,中间件的Invoke方法执行时就会存在使用的注入对象已经被释放的危险。所以,:Singleton依赖对象可以在中间件的构造函数中注入。
Scope 实列的获取应该在Invoke函数中获取。
namespace Nop.Services.Authentication
{
/// <summary>
/// Represents middleware that enables authentication
/// </summary>
public class AuthenticationMiddleware
{
...
public async Task Invoke(HttpContext context)
{
context.Features.Set<IAuthenticationFeature>(new AuthenticationFeature
{
OriginalPath = context.Request.Path,
OriginalPathBase = context.Request.PathBase
});
// Give any IAuthenticationRequestHandler schemes a chance to handle the request
var handlers = context.RequestServices.GetRequiredService<IAuthenticationHandlerProvider>();
await _next(context);
}
#endregion
}
}
source code:github
https://weblogs.asp.net/ricardoperes/asp-net-core-inversion-of-control-and-dependency-injection
Dependency injection in IHtmlHelper
public static IHtmlContent NopInlineScripts(this IHtmlHelper html, ResourceLocation location) //new
{
//var pageHeadBuilder = EngineContext.Current.Resolve<IPageHeadBuilder>();
var serviceProvider = html.ViewContext.HttpContext.RequestServices;
Custom 1、
public static void RegisterStandardTokens(this IHandlebars handlebars, IHttpContextAccessor httpContextAccessor)
{
handlebars.RegisterHelper("dateformat", (output, context, arguments) =>
{
var services = httpContextAccessor.HttpContext.RequestServices;
var clock = services.GetRequiredService<IClock>();
var siteService = services.GetRequiredService<ISiteService>();
var site = siteService.GetSiteSettingsAsync().GetAwaiter().GetResult();
var timeZone = TimeZoneInfo.FindSystemTimeZoneById(site.TimeZone);
var now = TimeZoneInfo.ConvertTime(clock.UtcNow, TimeZoneInfo.Utc, timeZone);
var format = arguments[0].ToString();
output.Write(now.ToString(format));
});
}
2、
/// <summary>
/// Add title element to the <![CDATA[<head>]]>
/// </summary>
/// <param name="html">HTML helper</param>
/// <param name="part">Title part</param>
public static void AddTitleParts(this IHtmlHelper htmlHelper, string part)
{
// var pageHeadBuilder = EngineContext.ServiceProvider.GetRequiredService<IPageHeadBuilder>();
var serviceProvider = htmlHelper.ViewContext.HttpContext.RequestServices;
var pageHeadBuilder = serviceProvider.GetRequiredService<IPageHeadBuilder>(); pageHeadBuilder.AddTitleParts(part);
}
3. public override string DisplayName
{
get
{
var services = new ServiceCollection();
var serviceProvider = services.BuildServiceProvider();
var logger = serviceProvider.GetRequiredService<IWorkContext>().WorkingLanguage.Id;
//get working language identifier
var workingLanguageId = EngineContext.ServiceProvider.GetRequiredService<IWorkContext>().WorkingLanguage.Id;
//get locale resource value
_resourceValue = EngineContext.ServiceProvider.GetRequiredService<ILocalizationService>().GetResource(ResourceKey);
return _resourceValue;
}
}
4.aspnet/AspNetCore – Startup.shared.cs
public async Task Auth(HttpContext ctx)
{
var authProvider = ctx.RequestServices.GetService<IAuthenticationSchemeProvider>();
var authScheme = (await authProvider.GetAllSchemesAsync()).SingleOrDefault();
await ctx.Response.WriteAsync(authScheme?.Name ?? "null");
if (ctx.User.Identity.Name != null)
{
await ctx.Response.WriteAsync(":" + ctx.User.Identity.Name);
}
}
5.IApplicationBuilder app
public void Configure(IApplicationBuilder app, IHostingEnvironment env, IApplicationLifetime appLifetime, ILoggerFactory loggerFactory)
using (var scope = app.ApplicationServices.CreateScope())
{
var initializer = scope.ServiceProvider.GetRequiredService<YourSampleDataInitializer>();
initializer.RunAsync().Wait();
}
6. Dependency injection where no httpcontext
7.
internal class ConsumeScopedServiceHostedService : IHostedService
{
private readonly ILogger _logger;
public ConsumeScopedServiceHostedService(IServiceProvider services,
ILogger<ConsumeScopedServiceHostedService> logger)
{
Services = services;
_logger = logger;
}
public IServiceProvider Services { get; }
public Task StartAsync(CancellationToken cancellationToken)
{
_logger.LogInformation(
"Consume Scoped Service Hosted Service is starting.");
DoWork();
return Task.CompletedTask;
}
private void DoWork()
{
_logger.LogInformation(
"Consume Scoped Service Hosted Service is working.");
using (var scope = Services.CreateScope())
{
var scopedProcessingService =
scope.ServiceProvider
.GetRequiredService<IScopedProcessingService>();
scopedProcessingService.DoWork();
}
}
public Task StopAsync(CancellationToken cancellationToken)
{
_logger.LogInformation(
"Consume Scoped Service Hosted Service is stopping.");
return Task.CompletedTask;
}
}
The services are registered in Startup.ConfigureServices. The IHostedService implementation is registered with the AddHostedService extension method:
services.AddHostedService<ConsumeScopedServiceHostedService>();
services.AddScoped<IScopedProcessingService, ScopedProcessingService>();
Introduction
There are quite a few good posts out there on Inversion of Control (IoC) and Dependency Injection (DI) in the ASP.NET Core world, but I felt there was still something to be said, hence this post! Mind you, this is going to be a long one! I once wrote another post on the history of dependency resolution in .NET, you may want to have a look at it. Always keep in mind that this is based on the latest bits, and may still change when it gets to the final version.
Services Registration
At bootstrap, ASP.NET Core will either call the ConfigureServices method, or one with a name following the convention Configure<environment>Services, where <environment> comes from the Hosting:Environment environment variable, and you can also set it in Visual Studio:
It is commonly set to “Development”, “Production”, “Staging”, etc. Keep in mind that if the environment-specific method exists (e.g., ConfigureDevelopmentServices), then the generic one (ConfigureServices) is not called.
The Configure*Services method is passed an instance of IServiceCollection, which is where we get to store our service registrations, so that they can be used by ASP.NET Core further along the pipeline. A service registration needs three things:
- A key type, normally an interface or an abstract base class;
- A concrete implementation of the key type;
- A lifetime.
The lifetime determines how many times the concrete implementation is going to be built. The ASP.NET Core service provider implementation knows about three different lifetimes:
- Scoped: an instance is created the first time it is requested during the same HTTP request, and during that request, this same instance is always returned;
- Singleton: an instance is created the first time it is requested, and this same instance is always returned in all subsequent calls;
- Transient: a new instance is always created, whenever it is requested.
A registration is an instance of the ServiceDescriptor class that is added to the IServiceCollection instance:
services.Add(new ServiceDescriptor(typeof(IMyService), typeof(MyService), ServiceLifetime.Singleton));
There are several extension methods that make this registration even easier.
What if you want to have custom processing when a service is returned? Simple:
services.AddTransient<IMyService>(sp =>
{
var something = sp.GetService(typeof(ISomething)) as ISomething;
var other = new OtherService();
//do something with it
return new MyService(other);
});
Keep in mind that the concrete service must either implement or inherit from the key service. Also, for the Scoped lifetime, if the concrete class implements IDisposable, the service provider will honor it and call Dispose at the end of the HTTP request.
After Configure*Services, ASP.NET Core calls Configure, also in the Startup class. This is where the core initialization occurs, like, adding all middleware, such as the MVC one, that makes the application works the way we expect it. The Configure method can either have no parameters or it can receive one or more parameters with types that match the registered services. For example, we will always have IApplicationBuilder, IHostingEnvironment, ILoggerFactory, so we can add them as parameters together with our own:
public void Configure(
IApplicationBuilder app,
IHostingEnvironment env,
ILoggerFactory loggerFactory,
IMyService service)
{
//do something with the services
}
It gives us a good chance to initialize them before the actual fun begins. If we want, we can also pass it just an instance of the dependency resolution service, represented by the IServiceProvider interface, which has been around since the early days of .NET. The ASP.NET Core service provider implements this interface, so having it here just means: “get me the service provider implementation so that I can retrieve services from it”:
public void Configure(IServiceProvider serviceProvider)
{
var env = serviceProvider.GetService(typeof(IHostingEnvironment)) as IHostingEnvironment;
var service = serviceProvider.GetService(typeof(IMyService)) as IMyService;
}
In the latest version of ASP.NET Core, there is no globally accessible reference to the built-in service provider, something known as the service locator, of bad reputation. If, for any reason you feel you need it, here is the place to store it. By the way, the Microsoft.Extensions.DependencyInjection package has some good strongly typed extension methods for IServiceProvider, make sure you include it.
Services
Each concrete service class itself can be dependency-injected: by default, it should be a concrete class and have a public parameter-less constructor; if, however, we want to inject dependencies into it, which also need to be registered in the service provider, we can have a constructor that takes these dependencies:
public class MyService : IMyService
{
public MyService(IServiceProvider serviceProvider)
{
//get something from the service provider
}
}
Notice the usage of IServiceProvider again. Another option is to have the constructor contain more concrete dependencies:
public class MyService : IMyService
{
public MyService(ILoggerFactory loggerFactory)
{
//do something with the logger factory
}
}
You can pass any number of parameters, of any type that is registered in the service provider, normally in one of the Configure*Services methods. If the specified service type is not registered, you get an exception at runtime.
Controllers
An MVC controller can have dependencies injected through its constructor, like I’ve shown for services:
public class HomeController : Controller
{
public HomeController(IMyService service)
{
//do something with the service
}
}
The same pattern applies: we can either declare a parameter as IServiceProvider or as a more specific type. It doesn’t matter if our controller inherits from Controller or is a POCO controller.
There was another dependency injection possibility with MVC controllers, which consisted in decorating public properties with a [FromServices] attribute: if a registration exists that matched the property’s type, it would be injected into the property just after the controller is constructed:
public class HomeController
{
[FromServices]
public IService Service { get; set; }
}
This is no longer possible, and the functionality was removed in DNX RC2, although it will still work in RC1. Forget about it.
However, you can still use [FromServices] in an action method:
public class HomeController : Controller
{
[HttpGet]
public IActionResult MyAction([FromServices] IService service) { return Ok(); }
}
By default, controller classes themselves do not come from the dependency resolution mechanism. You can achieve that, if, in Configure*Services, you call AddControllersAsServices, passing it either a Type or an Assembly collection:
services
.AddMvc()
.AddControllersAsServices(new[] { typeof(HomeController).GetTypeInfo().Assembly });
This allows you to do this:
services.AddSingleton<HomeController, SingletonHomeController>();
Each time MVC tries to build an HomeController, it will instead get the same SingletonHomeController. Note that I am not saying that you should do this, but I think you get the idea!
Filters
Filters in MVC allow you to intercept action method calls, results and exceptions, and do stuff before, after of instead of the real actions. You have global filters and attribute filters: global filters apply always, and attribute filters only apply in the scope where they are declared – a controller’s class or an action method. Filters themselves can have dependencies injected into them.
In order to declare global filters, you use one of the overloads of the AddMvc method, normally in the Configure*Servicesmethod, and registering the filter as a service:
services.AddMvc(mvc => mvc.Filters.AddService(typeof(GlobalFilter)));
MVC will try to resolve the filter type, if necessary, injecting it any dependencies, in the usual way:
public class GlobalFilter : IActionFilter
{
public GlobalFilter(IMyService service)
{
//do something with the service
}
public void OnActionExecuted(ActionExecutedContext context)
{
}
public void OnActionExecuting(ActionExecutingContext context)
{
}
}
The other option for filters is attributes, but attributes require a public parameter-less constructor. The MVC team came up with an alternative for that, in the form of the ServiceFilterAttribute and TypeFilterAttribute attributes. Both receive a Typeand the difference is that the first will try to resolve that type from the services registration and the last one will just instantiate it. Let’s see an example:
[ServiceFilter(typeof(GlobalFilter))]
public class HomeController : Controller
{
[TypeFilter(typeof(SomeFilter))]
public IActionResult Index()
{
//...
}
}
So, GlobalFilter is retrieved from the services registration, and, if it is a filter, it will behave accordingly to the context where it is located, in this case, it is the whole controller. SomeFilter, on the other hand, is just instantiated and used, no need to have it registered. Don’t forget that types passed to ServiceFilterAttribute and TypeFilterAttribute have to implement one of the filter interfaces in namespace Microsoft.AspNet.Mvc.Filters otherwise an exception is thrown.
Models
Model classes are declared as parameters to action methods in an MVC controller. There’s a binding mechanism that fills their properties automatically, but we can also use dependency injection here.
First, the whole model can come from dependency injection:
public IActionResult Index([FromServices] IMyService service)
{
//...
}
Or, at least, some properties of the model:
public IActionResult Update(Model model)
{
//...
}
public class Model
{
[FromServices]
public IMyService Service { get; set; }
//...
}
Views
MVC views can also take injected services, through the @inject directive:
@inject IMyService service;
<p>@service.Serve()</p>
View Components
View components, introduced in ASP.NET MVC Core, can also be injected in pretty much the same way as controllers:
public class MyServiceViewComponent : ViewComponent
{
public MyServiceViewComponent(IMyService service)
{
//do something with the service
}
public string Invoke()
{
//...
}
}
Tag Helpers
Tag helpers, like view components and controllers, also support constructor injection:
[HtmlTargetElement("service")]
public class ServiceTagHelper : TagHelper
{
public ServiceTagHelper(IMyService service)
{
//do something with the service
}
public override void Process(TagHelperContext context, TagHelperOutput output)
{
//...
}
}
Middleware
OWIN-style middleware are likewise injectable through the constructor:
public class ServiceMiddleware
{
private readonly RequestDelegate _next;
public ServiceMiddleware(RequestDelegate next, IMyService service)
{
this._next = next;
//do something with the service
}
public Task Invoke(HttpContext httpContext)
{
//...
}
}
Using a Custom IoC/DI Container
Now, you may not want to use the built-in service provider and use your own (Unity, Autofac, Ninject, TinyIoC, etc). That is certainly possible!
You need to change the signature of the Configure*Services method so as to return an IServiceProvider:
public IServiceProvider ConfigureServices(IServiceCollection services)
{
services.AddMvc();
services.AddTransient<IMyService, MyService>();
return new MyIoCContainer(services.BuildServiceProvider());
}
The key here is to create an instance of the service provider we want to use, maybe leveraging on the one produced from the service collection (BuildServiceProvider method), and return it here. Of course, if it has more features and lifetimes than the default one, you are free to use them all. The only contract it has to comply to is that of IServiceProvider.
Dependency Resolution
While running your MVC application, you can explicitly ask for services registered in the global (the default or your own) service provider. The HttpContext class exposes a RequestServices property; here you will find the custom service provider that you returned in Configure*Services, or ASP.NET Core’s built-in one. Whenever you have a reference to the current HttpContext (controllers, view components, tag helpers, views, middleware, filters), you get it as well. The “old” ApplicationServices was removed in recent commits, so it won’t make it to the final version of ASP.NET Core and you shouldn’t be using it.
Conclusion
The dependency injection mechanism was substantially changed in ASP.NET Core. The only identical feature seems to be constructor injection, and it is understandable, since it’s what most people should be using anyway. Now every moving part of ASP.NET Core is injectable through the same mechanism, which I think is a good thing. The problems so far have been the relative instability in the ASP.NET Core, things have been changing a lot, and there is still no light at the end of the tunnel.
https://www.cnblogs.com/artech/p/di-asp-net-core-pipeline-1.html
https://weblogs.asp.net/ricardoperes/asp-net-core-inversion-of-control-and-dependency-injection