Capita spesso di realizzare applicativi dei servizi in Blazor che richiedono un'inizializzazione asincrona - per esempio per recuperare dei parametri di configurazione tramite una chiamata HTTP.
Purtroppo, come sappiamo, non possiamo invocare metodi asincroni in un costruttore, quindi tipicamente la soluzione consiste nello scrivere codice boilerplate come il seguente, ossia creando un metodo che si occupi di inizializzare MyService, e che dobbiamo ricordarci di invocare in ognuno degli altri metodi che MyService espone.
public class MyService { private bool _isInitialized; public async Task Method1() { await this.EnsureInitializedAsync(); // Codice di Method1 } public async Task Method2() { await this.EnsureInitializedAsync(); // Codice di Method2 } public async Task EnsureInitializedAsync() { if (!_isInitialized) { // Codice di inizializzazione } } }
Ovviamente, oltre che scomoda, questa tecnica è anche soggetta a errori, perché potremmo dimenticarci la chiamata a EnsureInitializedAsync e introdurre bug nell'applicazione.
Una possibile alternativa, allora, è forzare l'inizializzazione di questa classe allo Startup, tramite il seguente codice:
public class Program { public static async Task Main(string[] args) { var builder = WebAssemblyHostBuilder.CreateDefault(args); builder.RootComponents.Add<App>("#app"); // registro i servizi su IoC container... builder.Services.AddScoped<MyService>(); var host = builder.Build(); var myService = host.Services.GetRequiredService<MyService>(); await myService.InitializeAsync(); await host.RunAsync(); } }
Dopo aver registrato il servizio, abbiamo rimpiazzato la tipica linea di codice
await builder.Build().RunAsync();
con una versione che dopo aver creato l'host, richieda un'istanza di MyService prima di chiamare RunAsync.
Essendo Main un metodo asincrono, possiamo effettuare l'await della nostra chiamata di inizializzazione. Inoltre, dato che questo codice viene eseguito prima che il componente App venga istanziato, siamo certi che nessun altro componente invochi MyService prima che questo sia pienamente operativo.
Commenti
Per inserire un commento, devi avere un account.
Fai il login e torna a questa pagina, oppure registrati alla nostra community.
Approfondimenti
Aggiungere interattività lato server in Blazor 8
Gestire domini wildcard in Azure Container Apps
Miglioramenti nelle performance di Angular 16
Evitare (o ridurre) il repo-jacking sulle GitHub Actions
Le novità di Angular: i miglioramenti alla CLI
Paginare i risultati con QuickGrid in Blazor
Ottenere il contenuto di una cartella FTP con la libreria FluentFTP
Sfruttare lo stream rendering per le pagine statiche di Blazor 8
Utilizzare Tailwind CSS all'interno di React: installazione
Miglioramenti agli screen reader e al contrasto in Angular
Eseguire attività pianificate con Azure Container Jobs
Definire stili a livello di libreria in Angular