--- title: ASP.NET Core Blazor dependency injection author: guardrex description: Learn how Blazor apps can inject services into components. monikerRange: '>= aspnetcore-3.1' ms.author: riande ms.custom: mvc ms.date: 11/12/2024 uid: blazor/fundamentals/dependency-injection --- # ASP.NET Core Blazor dependency injection [!INCLUDE[](~/includes/not-latest-version.md)] By [Rainer Stropek](https://www.timecockpit.com) and [Mike Rousos](https://github.com/mjrousos) This article explains how Blazor apps can inject services into components. [Dependency injection (DI)](xref:fundamentals/dependency-injection) is a technique for accessing services configured in a central location: * Framework-registered services can be injected directly into Razor components. * Blazor apps define and register custom services and make them available throughout the app via DI. > [!NOTE] > We recommend reading before reading this topic. ## Default services The services shown in the following table are commonly used in Blazor apps. | Service | Lifetime | Description | | ------- | -------- | ----------- | | | Scoped |

Provides methods for sending HTTP requests and receiving HTTP responses from a resource identified by a URI.

Client-side, an instance of is registered by the app in the `Program` file and uses the browser for handling the HTTP traffic in the background.

Server-side, an isn't configured as a service by default. In server-side code, provide an .

For more information, see .

An is registered as a scoped service, not singleton. For more information, see the [Service lifetime](#service-lifetime) section.

| | |

**Client-side**: Singleton

**Server-side**: Scoped

The Blazor framework registers in the app's service container.

|

Represents an instance of a JavaScript runtime where JavaScript calls are dispatched. For more information, see .

When seeking to inject the service into a singleton service on the server, take either of the following approaches:

  • Change the service registration to scoped to match 's registration, which is appropriate if the service deals with user-specific state.
  • Pass the into the singleton service's implementation as an argument of its method calls instead of injecting it into the singleton.
| | |

**Client-side**: Singleton

**Server-side**: Scoped

The Blazor framework registers in the app's service container.

| Contains helpers for working with URIs and navigation state. For more information, see [URI and navigation state helpers](xref:blazor/fundamentals/routing#uri-and-navigation-state-helpers). | Additional services registered by the Blazor framework are described in the documentation where they're used to describe Blazor features, such as configuration and logging. A custom service provider doesn't automatically provide the default services listed in the table. If you use a custom service provider and require any of the services shown in the table, add the required services to the new service provider. ## Add client-side services Configure services for the app's service collection in the `Program` file. In the following example, the `ExampleDependency` implementation is registered for `IExampleDependency`: ```csharp var builder = WebAssemblyHostBuilder.CreateDefault(args); ... builder.Services.AddSingleton(); ... await builder.Build().RunAsync(); ``` After the host is built, services are available from the root DI scope before any components are rendered. This can be useful for running initialization logic before rendering content: ```csharp var builder = WebAssemblyHostBuilder.CreateDefault(args); ... builder.Services.AddSingleton(); ... var host = builder.Build(); var weatherService = host.Services.GetRequiredService(); await weatherService.InitializeWeatherAsync(); await host.RunAsync(); ``` The host provides a central configuration instance for the app. Building on the preceding example, the weather service's URL is passed from a default configuration source (for example, `appsettings.json`) to `InitializeWeatherAsync`: ```csharp var builder = WebAssemblyHostBuilder.CreateDefault(args); ... builder.Services.AddSingleton(); ... var host = builder.Build(); var weatherService = host.Services.GetRequiredService(); await weatherService.InitializeWeatherAsync( host.Configuration["WeatherServiceUrl"]); await host.RunAsync(); ``` ## Add server-side services After creating a new app, examine part of the `Program` file: :::moniker range=">= aspnetcore-8.0" ```csharp var builder = WebApplication.CreateBuilder(args); builder.Services.AddRazorComponents() .AddInteractiveServerComponents(); ``` :::moniker-end :::moniker range="< aspnetcore-8.0" ```csharp var builder = WebApplication.CreateBuilder(args); builder.Services.AddRazorPages(); builder.Services.AddServerSideBlazor(); builder.Services.AddSingleton(); ``` :::moniker-end The `builder` variable represents a with an , which is a list of [service descriptor](xref:Microsoft.Extensions.DependencyInjection.ServiceDescriptor) objects. Services are added by providing service descriptors to the service collection. The following example demonstrates the concept with the `IDataAccess` interface and its concrete implementation `DataAccess`: ```csharp builder.Services.AddSingleton(); ``` :::moniker range="< aspnetcore-6.0" After creating a new app, examine the `Startup.ConfigureServices` method in `Startup.cs`: ```csharp using Microsoft.Extensions.DependencyInjection; ... public void ConfigureServices(IServiceCollection services) { ... } ``` The method is passed an , which is a list of [service descriptor](xref:Microsoft.Extensions.DependencyInjection.ServiceDescriptor) objects. Services are added in the `ConfigureServices` method by providing service descriptors to the service collection. The following example demonstrates the concept with the `IDataAccess` interface and its concrete implementation `DataAccess`: ```csharp public void ConfigureServices(IServiceCollection services) { services.AddSingleton(); } ``` :::moniker-end ## Register common services If one or more common services are required client- and server-side, you can place the common service registrations in a method client-side and call the method to register the services in both projects. First, factor common service registrations into a separate method. For example, create a `ConfigureCommonServices` method client-side: ```csharp public static void ConfigureCommonServices(IServiceCollection services) { services.Add...; } ``` For the client-side `Program` file, call `ConfigureCommonServices` to register the common services: ```csharp var builder = WebAssemblyHostBuilder.CreateDefault(args); ... ConfigureCommonServices(builder.Services); ``` In the server-side `Program` file, call `ConfigureCommonServices` to register the common services: ```csharp var builder = WebApplication.CreateBuilder(args); ... Client.Program.ConfigureCommonServices(builder.Services); ``` For an example of this approach, see . :::moniker range=">= aspnetcore-8.0" ## Client-side services that fail during prerendering *This section only applies to WebAssembly components in Blazor Web Apps.* Blazor Web Apps normally prerender client-side WebAssembly components. If an app is run with a required service only registered in the `.Client` project, executing the app results in a runtime error similar to the following when a component attempts to use the required service during prerendering: > :::no-loc text="InvalidOperationException: Cannot provide a value for {PROPERTY} on type '{ASSEMBLY}}.Client.Pages.{COMPONENT NAME}'. There is no registered service of type '{SERVICE}'."::: Use ***either*** of the following approaches to resolve this problem: * Register the service in the main project to make it available during component prerendering. * If prerendering isn't required for the component, disable prerendering by following the guidance in . If you adopt this approach, you don't need to register the service in the main project. For more information, see [Client-side services fail to resolve during prerendering](xref:blazor/components/render-modes#client-side-services-fail-to-resolve-during-prerendering). :::moniker-end ## Service lifetime Services can be configured with the lifetimes shown in the following table. | Lifetime | Description | | -------- | ----------- | | |

Client-side doesn't currently have a concept of DI scopes. `Scoped`-registered services behave like `Singleton` services.

Server-side development supports the `Scoped` lifetime across HTTP requests but not across SignalR connection/circuit messages among components that are loaded on the client. The Razor Pages or MVC portion of the app treats scoped services normally and recreates the services on *each HTTP request* when navigating among pages or views or from a page or view to a component. Scoped services aren't reconstructed when navigating among components on the client, where the communication to the server takes place over the SignalR connection of the user's circuit, not via HTTP requests. In the following component scenarios on the client, scoped services are reconstructed because a new circuit is created for the user:

  • The user closes the browser's window. The user opens a new window and navigates back to the app.
  • The user closes a tab of the app in a browser window. The user opens a new tab and navigates back to the app.
  • The user selects the browser's reload/refresh button.

For more information on preserving user state in server-side apps, see .

| | | DI creates a *single instance* of the service. All components requiring a `Singleton` service receive the same instance of the service. | | | Whenever a component obtains an instance of a `Transient` service from the service container, it receives a *new instance* of the service. | The DI system is based on the DI system in ASP.NET Core. For more information, see . ## Request a service in a component :::moniker range=">= aspnetcore-9.0" For injecting services into components, Blazor supports [constructor injection](#constructor-injection) and [property injection](#property-injection). ### Constructor injection After services are added to the service collection, inject one or more services into components with constructor injection. The following example injects the `NavigationManager` service. `ConstructorInjection.razor`: ```razor @page "/constructor-injection" ``` `ConstructorInjection.razor.cs`: ```csharp using Microsoft.AspNetCore.Components; public partial class ConstructorInjection(NavigationManager navigation) { private void HandleClick() { navigation.NavigateTo("/counter"); } } ``` ### Property injection :::moniker-end After services are added to the service collection, inject one or more services into components with the [`@inject`](xref:mvc/views/razor#inject) Razor directive, which has two parameters: * Type: The type of the service to inject. * Property: The name of the property receiving the injected app service. The property doesn't require manual creation. The compiler creates the property. For more information, see . Use multiple [`@inject`](xref:mvc/views/razor#inject) statements to inject different services. The following example demonstrates shows how to use the [`@inject`](xref:mvc/views/razor#inject) directive. The service implementing `Services.NavigationManager` is injected into the component's property `Navigation`. Note how the code is only using the `NavigationManager` abstraction. `PropertyInjection.razor`: ```razor @page "/property-injection" @inject NavigationManager Navigation ``` Internally, the generated property (`Navigation`) uses the [`[Inject]` attribute](xref:Microsoft.AspNetCore.Components.InjectAttribute). Typically, this attribute isn't used directly. If a base class is required for components and injected properties are also required for the base class, manually add the [`[Inject]` attribute](xref:Microsoft.AspNetCore.Components.InjectAttribute): ```csharp using Microsoft.AspNetCore.Components; public class ComponentBase : IComponent { [Inject] protected NavigationManager Navigation { get; set; } = default!; ... } ``` > [!NOTE] > Since injected services are expected to be available, the default literal with the null-forgiving operator (`default!`) is assigned in .NET 6 or later. For more information, see [Nullable reference types (NRTs) and .NET compiler null-state static analysis](xref:migration/50-to-60#nullable-reference-types-nrts-and-net-compiler-null-state-static-analysis). In components derived from a base class, the [`@inject`](xref:mvc/views/razor#inject) directive isn't required. The of the base class is sufficient. The component only requires the [`@inherits`](xref:mvc/views/razor#inherits) directive. In the following example, any injected services of `CustomComponentBase` are available to the `Demo` component: ```razor @page "/demo" @inherits CustomComponentBase ``` ## Use DI in services Complex services might require additional services. In the following example, `DataAccess` requires the default service. [`@inject`](xref:mvc/views/razor#inject) (or the [`[Inject]` attribute](xref:Microsoft.AspNetCore.Components.InjectAttribute)) isn't available for use in services. *Constructor injection* must be used instead. Required services are added by adding parameters to the service's constructor. When DI creates the service, it recognizes the services it requires in the constructor and provides them accordingly. In the following example, the constructor receives an via DI. is a default service. ```csharp using System.Net.Http; public class DataAccess : IDataAccess { public DataAccess(HttpClient http) { ... } ... } ``` Constructor injection is supported with [primary constructors](/dotnet/csharp/whats-new/tutorials/primary-constructors) in C# 12 (.NET 8) or later: ```csharp using System.Net.Http; public class DataAccess(HttpClient http) : IDataAccess { ... } ``` Prerequisites for constructor injection: * One constructor must exist whose arguments can all be fulfilled by DI. Additional parameters not covered by DI are allowed if they specify default values. * The applicable constructor must be `public`. * One applicable constructor must exist. In case of an ambiguity, DI throws an exception. :::moniker range=">= aspnetcore-8.0" ## Inject keyed services into components Blazor supports injecting keyed services using the `[Inject]` attribute. Keys allow for scoping of registration and consumption of services when using dependency injection. Use the property to specify the key for the service to inject: ```csharp [Inject(Key = "my-service")] public IMyService MyService { get; set; } ``` :::moniker-end ## Utility base component classes to manage a DI scope In non-Blazor ASP.NET Core apps, scoped and transient services are typically scoped to the current request. After the request completes, scoped and transient services are disposed by the DI system. In interactive server-side Blazor apps, the DI scope lasts for the duration of the circuit (the SignalR connection between the client and server), which can result in scoped and disposable transient services living much longer than the lifetime of a single component. Therefore, don't directly inject a scoped service into a component if you intend the service lifetime to match the lifetime of the component. Transient services injected into a component that don't implement are garbage collected when the component is disposed. However, injected transient services *that implement * are maintained by the DI container for the lifetime of the circuit, which prevents service garbage collection when the component is disposed and results in a memory leak. An alternative approach for scoped services based on the type is described later in this section, and disposable transient services shouldn't be used at all. For more information, see [Design for solving transient disposables on Blazor Server (`dotnet/aspnetcore` #26676)](https://github.com/dotnet/aspnetcore/issues/26676). Even in client-side Blazor apps that don't operate over a circuit, services registered with a scoped lifetime are treated as singletons, so they live longer than scoped services in typical ASP.NET Core apps. Client-side disposable transient services also live longer than the components where they're injected because the DI container, which holds references to disposable services, persists for the lifetime of the app, preventing garbage collection on the services. Although long-lived disposable transient services are of greater concern on the server, they should be avoided as client service registrations as well. Use of the type is also recommended for client-side scoped services to control service lifetime, and disposable transient services shouldn't be used at all. An approach that limits a service lifetime is use of the type. is an abstract type derived from that creates a DI scope corresponding to the *lifetime of the component*. Using this scope, a component can inject services with a scoped lifetime and have them live as long as the component. When the component is destroyed, services from the component's scoped service provider are disposed as well. This can be useful for services reused within a component but not shared across components. Two versions of type are available and described in the next two sections: * [`OwningComponentBase`](#owningcomponentbase) * [`OwningComponentBase`](#owningcomponentbasetservice) ### `OwningComponentBase` is an abstract, disposable child of the type with a protected property of type . The provider can be used to resolve services that are scoped to the lifetime of the component. DI services injected into the component using [`@inject`](xref:mvc/views/razor#inject) or the [`[Inject]` attribute](xref:Microsoft.AspNetCore.Components.InjectAttribute) aren't created in the component's scope. To use the component's scope, services must be resolved using with either or . Any services resolved using the provider have their dependencies provided in the component's scope. The following example demonstrates the difference between injecting a scoped service directly and resolving a service using on the server. The following interface and implementation for a time travel class include a `DT` property to hold a value. The implementation calls to set `DT` when the `TimeTravel` class is instantiated. `ITimeTravel.cs`: ```csharp public interface ITimeTravel { public DateTime DT { get; set; } } ``` `TimeTravel.cs`: ```csharp public class TimeTravel : ITimeTravel { public DateTime DT { get; set; } = DateTime.Now; } ``` The service is registered as scoped in the server-side `Program` file. Server-side, scoped services have a lifetime equal to the duration of the [circuit](xref:blazor/hosting-models#blazor-server). In the `Program` file: ```csharp builder.Services.AddScoped(); ``` In the following `TimeTravel` component: * The time travel service is directly injected with `@inject` as `TimeTravel1`. * The service is also resolved separately with and as `TimeTravel2`. `TimeTravel.razor`: :::moniker range=">= aspnetcore-8.0" ```razor @page "/time-travel" @inject ITimeTravel TimeTravel1 @inherits OwningComponentBase

OwningComponentBase Example

  • TimeTravel1.DT: @TimeTravel1?.DT
  • TimeTravel2.DT: @TimeTravel2?.DT
@code { private ITimeTravel TimeTravel2 { get; set; } = default!; protected override void OnInitialized() { TimeTravel2 = ScopedServices.GetRequiredService(); } } ``` :::moniker-end :::moniker range="< aspnetcore-8.0" ```razor @page "/time-travel" @inject ITimeTravel TimeTravel1 @inherits OwningComponentBase

OwningComponentBase Example

  • TimeTravel1.DT: @TimeTravel1?.DT
  • TimeTravel2.DT: @TimeTravel2?.DT
@code { private ITimeTravel TimeTravel2 { get; set; } = default!; protected override void OnInitialized() { TimeTravel2 = ScopedServices.GetRequiredService(); } } ``` :::moniker-end Initially navigating to the `TimeTravel` component, the time travel service is instantiated twice when the component loads, and `TimeTravel1` and `TimeTravel2` have the same initial value: > :::no-loc text="TimeTravel1.DT: 8/31/2022 2:54:45 PM"::: > :::no-loc text="TimeTravel2.DT: 8/31/2022 2:54:45 PM"::: When navigating away from the `TimeTravel` component to another component and back to the `TimeTravel` component: * `TimeTravel1` is provided the same service instance that was created when the component first loaded, so the value of `DT` remains the same. * `TimeTravel2` obtains a new `ITimeTravel` service instance in `TimeTravel2` with a new DT value. > :::no-loc text="TimeTravel1.DT: 8/31/2022 2:54:45 PM"::: > :::no-loc text="TimeTravel2.DT: 8/31/2022 2:54:48 PM"::: `TimeTravel1` is tied to the user's circuit, which remains intact and isn't disposed until the underlying circuit is deconstructed. For example, the service is disposed if the circuit is disconnected for the [disconnected circuit retention period](xref:Microsoft.AspNetCore.Components.Server.CircuitOptions.DisconnectedCircuitRetentionPeriod). In spite of the scoped service registration in the `Program` file and the longevity of the user's circuit, `TimeTravel2` receives a new `ITimeTravel` service instance each time the component is initialized. ### `OwningComponentBase` derives from and adds a property that returns an instance of `T` from the scoped DI provider. This type is a convenient way to access scoped services without using an instance of when there's one primary service the app requires from the DI container using the component's scope. The property is available, so the app can get services of other types, if necessary. ```razor @page "/users" @attribute [Authorize] @inherits OwningComponentBase

Users (@Service.Users.Count())

    @foreach (var user in Service.Users) {
  • @user.UserName
  • }
``` :::moniker range=">= aspnetcore-6.0" ### Detect client-side transient disposables Custom code can be added to a client-side Blazor app to detect disposable transient services in an app that should use . This approach is useful if you're concerned that code added to the app in the future consumes one or more transient disposable services, including services added by libraries. Demonstration code is available in the [Blazor samples GitHub repository](https://github.com/dotnet/blazor-samples/tree/main) ([how to download](xref:blazor/fundamentals/index#sample-apps)). Inspect the following in .NET 6 or later versions of the `BlazorSample_WebAssembly` sample: * `DetectIncorrectUsagesOfTransientDisposables.cs` * `Services/TransientDisposableService.cs` * In `Program.cs`: * The app's `Services` namespace is provided at the top of the file (`using BlazorSample.Services;`). * `DetectIncorrectUsageOfTransients` is called immediately after the `builder` is assigned from . * The `TransientDisposableService` is registered (`builder.Services.AddTransient();`). * `EnableTransientDisposableDetection` is called on the built host in the processing pipeline of the app (`host.EnableTransientDisposableDetection();`). * The app registers the `TransientDisposableService` service without throwing an exception. However, attempting to resolve the service in `TransientService.razor` throws an when the framework attempts to construct an instance of `TransientDisposableService`. ### Detect server-side transient disposables Custom code can be added to a server-side Blazor app to detect server-side disposable transient services in an app that should use . This approach is useful if you're concerned that code added to the app in the future consumes one or more transient disposable services, including services added by libraries. Demonstration code is available in the [Blazor samples GitHub repository](https://github.com/dotnet/blazor-samples/tree/main) ([how to download](xref:blazor/fundamentals/index#sample-apps)). :::moniker-end :::moniker range=">= aspnetcore-8.0" Inspect the following in .NET 8 or later versions of the `BlazorSample_BlazorWebApp` sample: :::moniker-end :::moniker range=">= aspnetcore-6.0 < aspnetcore-8.0" Inspect the following in .NET 6 or .NET 7 versions of the `BlazorSample_Server` sample: :::moniker-end :::moniker range=">= aspnetcore-6.0" * `DetectIncorrectUsagesOfTransientDisposables.cs` * `Services/TransitiveTransientDisposableDependency.cs`: * In `Program.cs`: * The app's `Services` namespace is provided at the top of the file (`using BlazorSample.Services;`). * `DetectIncorrectUsageOfTransients` is called on the host builder (`builder.DetectIncorrectUsageOfTransients();`). * The `TransientDependency` service is registered (`builder.Services.AddTransient();`). * The `TransitiveTransientDisposableDependency` is registered for `ITransitiveTransientDisposableDependency` (`builder.Services.AddTransient();`). * The app registers the `TransientDependency` service without throwing an exception. However, attempting to resolve the service in `TransientService.razor` throws an when the framework attempts to construct an instance of `TransientDependency`. ### Transient service registrations for `IHttpClientFactory`/`HttpClient` handlers Transient service registrations for / handlers are recommended. If the app contains / handlers and uses the to add support for authentication, the following transient disposables for client-side authentication are also discovered, which is expected and can be ignored: * * Other instances of / are also discovered. These instances can also be ignored. The Blazor sample apps in the [Blazor samples GitHub repository](https://github.com/dotnet/blazor-samples/tree/main) ([how to download](xref:blazor/fundamentals/index#sample-apps)) demonstrate the code to detect transient disposables. However, the code is deactivated because the sample apps include / handlers. To activate the demonstration code and witness its operation: * Uncomment the transient disposable lines in `Program.cs`. * Remove the conditional check in `NavLink.razor` that prevents the `TransientService` component from displaying in the app's navigation sidebar: ```diff - else if (name != "TransientService") + else ``` * Run the sample app and navigate to the `TransientService` component at `/transient-service`. :::moniker-end ## Use of an Entity Framework Core (EF Core) DbContext from DI For more information, see . ## Access server-side Blazor services from a different DI scope :::moniker range=">= aspnetcore-8.0" [Circuit activity handlers](xref:blazor/fundamentals/signalr#monitor-server-side-circuit-activity) provide an approach for accessing scoped Blazor services from other non-Blazor dependency injection (DI) scopes, such as scopes created using . Prior to the release of ASP.NET Core in .NET 8, accessing circuit-scoped services from other dependency injection scopes required using a custom base component type. With circuit activity handlers, a custom base component type isn't required, as the following example demonstrates: ```csharp public class CircuitServicesAccessor { static readonly AsyncLocal blazorServices = new(); public IServiceProvider? Services { get => blazorServices.Value; set => blazorServices.Value = value; } } public class ServicesAccessorCircuitHandler( IServiceProvider services, CircuitServicesAccessor servicesAccessor) : CircuitHandler { public override Func CreateInboundActivityHandler( Func next) => async context => { servicesAccessor.Services = services; await next(context); servicesAccessor.Services = null; }; } public static class CircuitServicesServiceCollectionExtensions { public static IServiceCollection AddCircuitServicesAccessor( this IServiceCollection services) { services.AddScoped(); services.AddScoped(); return services; } } ``` Access the circuit-scoped services by injecting the `CircuitServicesAccessor` where it's needed. For an example that shows how to access the from a set up using , see . :::moniker-end :::moniker range="< aspnetcore-8.0" There may be times when a Razor component invokes asynchronous methods that execute code in a different DI scope. Without the correct approach, these DI scopes don't have access to Blazor's services, such as and . For example, instances created using have their own DI service scope. As a result, instances configured on the aren't able to directly inject Blazor services. Create a class `BlazorServiceAccessor` that defines an [`AsyncLocal`](xref:System.Threading.AsyncLocal`1), which stores the Blazor for the current asynchronous context. A `BlazorServiceAccessor` instance can be acquired from within a different DI service scope to access Blazor services. `BlazorServiceAccessor.cs`: ```csharp internal sealed class BlazorServiceAccessor { private static readonly AsyncLocal s_currentServiceHolder = new(); public IServiceProvider? Services { get => s_currentServiceHolder.Value?.Services; set { if (s_currentServiceHolder.Value is { } holder) { // Clear the current IServiceProvider trapped in the AsyncLocal. holder.Services = null; } if (value is not null) { // Use object indirection to hold the IServiceProvider in an AsyncLocal // so it can be cleared in all ExecutionContexts when it's cleared. s_currentServiceHolder.Value = new() { Services = value }; } } } private sealed class BlazorServiceHolder { public IServiceProvider? Services { get; set; } } } ``` To set the value of `BlazorServiceAccessor.Services` automatically when an `async` component method is invoked, create a custom base component that re-implements the three primary asynchronous entry points into Razor component code: * * * The following class demonstrates the implementation for the base component. `CustomComponentBase.cs`: ```csharp using Microsoft.AspNetCore.Components; public class CustomComponentBase : ComponentBase, IHandleEvent, IHandleAfterRender { private bool hasCalledOnAfterRender; [Inject] private IServiceProvider Services { get; set; } = default!; [Inject] private BlazorServiceAccessor BlazorServiceAccessor { get; set; } = default!; public override Task SetParametersAsync(ParameterView parameters) => InvokeWithBlazorServiceContext(() => base.SetParametersAsync(parameters)); Task IHandleEvent.HandleEventAsync(EventCallbackWorkItem callback, object? arg) => InvokeWithBlazorServiceContext(() => { var task = callback.InvokeAsync(arg); var shouldAwaitTask = task.Status != TaskStatus.RanToCompletion && task.Status != TaskStatus.Canceled; StateHasChanged(); return shouldAwaitTask ? CallStateHasChangedOnAsyncCompletion(task) : Task.CompletedTask; }); Task IHandleAfterRender.OnAfterRenderAsync() => InvokeWithBlazorServiceContext(() => { var firstRender = !hasCalledOnAfterRender; hasCalledOnAfterRender |= true; OnAfterRender(firstRender); return OnAfterRenderAsync(firstRender); }); private async Task CallStateHasChangedOnAsyncCompletion(Task task) { try { await task; } catch { if (task.IsCanceled) { return; } throw; } StateHasChanged(); } private async Task InvokeWithBlazorServiceContext(Func func) { try { BlazorServiceAccessor.Services = Services; await func(); } finally { BlazorServiceAccessor.Services = null; } } } ``` Any components extending `CustomComponentBase` automatically have `BlazorServiceAccessor.Services` set to the in the current Blazor DI scope. :::moniker-end :::moniker range=">= aspnetcore-6.0 < aspnetcore-8.0" Finally, in the `Program` file, add the `BlazorServiceAccessor` as a scoped service: ```csharp builder.Services.AddScoped(); ``` :::moniker-end :::moniker range="< aspnetcore-6.0" Finally, in `Startup.ConfigureServices` of `Startup.cs`, add the `BlazorServiceAccessor` as a scoped service: ```csharp services.AddScoped(); ``` :::moniker-end ## Additional resources :::moniker range=">= aspnetcore-8.0" * [Service injection via a top-level imports file (`_Imports.razor`) in Blazor Web Apps](xref:blazor/components/render-modes#service-injection-via-a-top-level-imports-file-_importsrazor) * * [`IDisposable` guidance for Transient and shared instances](xref:fundamentals/dependency-injection#idisposable-guidance-for-transient-and-shared-instances) * * [Primary constructors (C# Guide)](/dotnet/csharp/programming-guide/classes-and-structs/instance-constructors#primary-constructors) :::moniker-end :::moniker range="< aspnetcore-8.0" * * [`IDisposable` guidance for Transient and shared instances](xref:fundamentals/dependency-injection#idisposable-guidance-for-transient-and-shared-instances) * :::moniker-end