Partage via


Prérendu des composants ASP.NET Core Razor

Remarque

Ceci n’est pas la dernière version de cet article. Pour la version actuelle, consultez la version .NET 9 de cet article.

Important

Ces informations portent sur la préversion du produit, qui est susceptible d’être en grande partie modifié avant sa commercialisation. Microsoft n’offre aucune garantie, expresse ou implicite, concernant les informations fournies ici.

Pour la version actuelle, consultez la version .NET 9 de cet article.

Cet article explique les scénarios de prérendu des composants Razor pour les composants rendus par serveur dans les Blazor Web App.

Le prérendu est le processus de rendu initial du contenu d’une page sur le serveur sans activation des gestionnaires d’événements pour les contrôles rendus. Le serveur génère l’interface utilisateur HTML de la page dès que possible en réponse à la demande initiale, ce qui rend l’application plus réactive pour les utilisateurs. Le prérendu peut aussi améliorer l’optimisation du référencement d’un site auprès d’un moteur de recherche (SEO) en rendant le contenu de la réponse HTTP initiale qui est utilisée par les moteurs de recherche pour calculer le rang de la page.

Conserver l’état prérendu

Si l’état utilisé durant le prérendu n’est pas conservé, il est perdu et doit être recréé lorsque l’application est entièrement chargée. Si un état est créé de façon asynchrone, l’interface utilisateur peut scintiller pendant que l’IU prérendue est remplacée alors que le composant régénère le rendu.

Considérez le compteur de composants PrerenderedCounter1 suivant. Le composant définit une valeur de compteur aléatoire initiale pendant un prérendu dans une méthode de cycle de vie OnInitialized. Une fois la connexion SignalR avec le client établie, le composant régénère le rendu et la valeur du compteur initiale est remplacée lorsque OnInitialized s’exécute une seconde fois.

PrerenderedCounter1.razor:

@page "/prerendered-counter-1"
@rendermode @(new InteractiveServerRenderMode(prerender: true))
@inject ILogger<PrerenderedCounter1> Logger

<PageTitle>Prerendered Counter 1</PageTitle>

<h1>Prerendered Counter 1</h1>

<p role="status">Current count: @currentCount</p>

<button class="btn btn-primary" @onclick="IncrementCount">Click me</button>

@code {
    private int currentCount;

    protected override void OnInitialized()
    {
        currentCount = Random.Shared.Next(100);
        Logger.LogInformation("currentCount set to {Count}", currentCount);
    }

    private void IncrementCount() => currentCount++;
}

Exécutez l’application et inspectez la journalisation à partir du composant. Voici un exemple de sortie.

Remarque

Si l'application adopte le routage interactif et que la page est atteinte via une navigation interne améliorée , le prérendering ne se produit pas. Par conséquent, vous devez effectuer un rechargement de page complet pour que le composant PrerenderedCounter1 affiche la sortie suivante. Pour plus d'informations, voir la section Routage interactif et pré-rendu.

info: BlazorSample.Components.Pages.PrerenderedCounter1[0]
currentCount set to 41
info: BlazorSample.Components.Pages.PrerenderedCounter1[0]
currentCount set to 92

Le premier compte journalisé a lieu lors du prérendu. Le compte est à nouveau défini après le prérendu lorsque le composant régénère le rendu. Un scintillement dans l’interface utilisateur est également possible lorsque le compte passe de 41 à 92.

Pour conserver la valeur initiale du compteur lors du prérendu, Blazor prend en charge l’état persistant dans une page prérendue à l’aide du service PersistentComponentState (et pour les composants incorporés dans des pages ou des vues d’applications Razor Pages ou MVC, via le Tag Helper d’état du composant persistant).

Pour conserver l’état prérendu, décidez de l’état à conserver à l’aide du service PersistentComponentState. PersistentComponentState.RegisterOnPersisting enregistre un rappel pour conserver l’état du composant avant que l’application ne soit suspendue. L’état est récupéré lorsque l’application reprend.

L’exemple suivant illustre le modèle général :

  • L’espace réservé {TYPE} représente le type de données à conserver.
  • L’espace réservé {TOKEN} est une chaîne de l’identificateur d’état. Envisagez d’utiliser nameof({VARIABLE}), où l’espace réservé {VARIABLE} est le nom de la variable qui contient l’état. L’utilisation de nameof() pour l’identifiant de l’État permet d’éviter l’utilisation d’une chaîne de caractères entre guillemets.
@implements IDisposable
@inject PersistentComponentState ApplicationState

...

@code {
    private {TYPE} data;
    private PersistingComponentStateSubscription persistingSubscription;

    protected override async Task OnInitializedAsync()
    {
        persistingSubscription = 
            ApplicationState.RegisterOnPersisting(PersistData);

        if (!ApplicationState.TryTakeFromJson<{TYPE}>(
            "{TOKEN}", out var restored))
        {
            data = await ...;
        }
        else
        {
            data = restored!;
        }
    }

    private Task PersistData()
    {
        ApplicationState.PersistAsJson("{TOKEN}", data);

        return Task.CompletedTask;
    }

    void IDisposable.Dispose()
    {
        persistingSubscription.Dispose();
    }
}

Le composant compteur suivant conserve l’état du compteur pendant le prérendu, puis récupère l’état pour initialiser le composant.

PrerenderedCounter2.razor:

@page "/prerendered-counter-2"
@implements IDisposable
@inject ILogger<PrerenderedCounter2> Logger
@inject PersistentComponentState ApplicationState

<PageTitle>Prerendered Counter 2</PageTitle>

<h1>Prerendered Counter 2</h1>

<p role="status">Current count: @currentCount</p>

<button class="btn btn-primary" @onclick="IncrementCount">Click me</button>

@code {
    private int currentCount;
    private PersistingComponentStateSubscription persistingSubscription;

    protected override void OnInitialized()
    {
        persistingSubscription =
            ApplicationState.RegisterOnPersisting(PersistCount);

        if (!ApplicationState.TryTakeFromJson<int>(
            nameof(currentCount), out var restoredCount))
        {
            currentCount = Random.Shared.Next(100);
            Logger.LogInformation("currentCount set to {Count}", currentCount);
        }
        else
        {
            currentCount = restoredCount!;
            Logger.LogInformation("currentCount restored to {Count}", currentCount);
        }
    }

    private Task PersistCount()
    {
        ApplicationState.PersistAsJson(nameof(currentCount), currentCount);

        return Task.CompletedTask;
    }

    void IDisposable.Dispose() => persistingSubscription.Dispose();

    private void IncrementCount() => currentCount++;
}

Lorsque le composant s’exécute, currentCount n’est défini que lors du prérendu. La valeur est restaurée lors du nouveau rendu du composant. Voici un exemple de sortie.

Remarque

Si l'application adopte le routage interactif et que la page est atteinte via une navigation interne améliorée, le prerendering ne se produit pas. Par conséquent, vous devez effectuer un rechargement de page complet pour que le composant PrerenderedCounter2 affiche la sortie suivante. Pour plus d'informations, voir la section Routage interactif et pré-rendu.

info: BlazorSample.Components.Pages.PrerenderedCounter2[0]
currentCount set to 96
info: BlazorSample.Components.Pages.PrerenderedCounter2[0]
currentCount restored to 96

En initialisant des composants avec le même état que celui utilisé durant le prérendu, toutes les étapes d’initialisation coûteuses ne sont exécutées qu’une seule fois. L’interface utilisateur rendue correspond également à l’interface utilisateur prérendue, de sorte qu’aucun scintillement ne se produit dans le navigateur.

L’état préréenderé persistant est transféré au client, où il est utilisé pour restaurer l’état du composant. Pendant le rendu côté client (CSR), InteractiveWebAssemblyles données sont exposées au navigateur et ne doivent pas contenir d’informations sensibles et privées. Pendant le rendu interactif côté serveur (SSR interactif), InteractiveServerASP.NET Core Data Protection garantit que les données sont transférées en toute sécurité. Le InteractiveAuto mode de rendu combine l’interactivité WebAssembly et Server. Il est donc nécessaire de prendre en compte l’exposition des données au navigateur, comme dans le cas de la demande de signature de certificat.

Composants incorporés dans des pages et des vues (MVC/Pages Razor)

Pour les composants incorporés dans une page ou une vue d’une application Razor Pages ou MVC, vous devez ajouter le Tag Helper d’état du composant persistant avec la balise HTML <persist-component-state /> à l’intérieur de la balise fermante </body> du layout de l’application. Cela n’est nécessaire que pour les applications Razor Pages et MVC. Pour plus d’informations, consultez l’assistance Tag Helper d’état du composant persistant dans ASP.NET Core.

Pages/Shared/_Layout.cshtml:

<body>
    ...

    <persist-component-state />
</body>

Routage interactif et prérendu

Lorsque le composant Routes ne définit pas de mode de rendu, l’application utilise l’interactivité et la navigation par page/composant. En utilisant la navigation par page/composant, la navigation interne† est gérée par le routage amélioré après que l'application est devenue interactive. †Interne dans ce contexte signifie que la destination URL de l'événement de navigation est un point de terminaison Blazor à l'intérieur de l'application.

Le service PersistentComponentState fonctionne uniquement sur le chargement initial de la page et non sur les événements de navigation de page améliorés internes.

Si l’application effectue une navigation complète (non améliorée) vers une page utilisant l’état du composant persistant, l’état persistant est mis à la disposition de l’application lorsqu’elle devient interactive.

Si un circuit interactif a déjà été établi et qu’une navigation améliorée est effectuée sur une page utilisant l’état du composant persistant, l’état n’est pas disponible dans le circuit existant pour que le composant utilise. Il n'y a pas de prérendu pour la requête de page interne, et le service PersistentComponentState n'est pas au courant qu'une navigation améliorée s'est produite. Il n’existe aucun mécanisme permettant de fournir des mises à jour d’état aux composants qui s’exécutent déjà sur un circuit existant. La raison en est que Blazor prend uniquement en charge le passage de l’état du serveur au client au moment de l’initialisation du runtime, et non après le démarrage du runtime.

Des travaux supplémentaires sur le framework de Blazor pour résoudre ce scénario sont pris en considération pour .NET 10 (novembre 2025). Pour plus d'informations et une discussion de la communauté sur les solutions de contournement non prises en charge‡, voir Support persistent component state across enhanced page navigations ( dotnet/aspnetcore51584). Les solutions de contournement non prises en charge ne sont pas approuvées par Microsoft pour une utilisation dans les applications Blazor. Utiliser des packages, des approches et du code tiers à vos propres risques.

La désactivation de la navigation améliorée, qui réduit les performances mais évite également le problème du chargement de l'état avec PersistentComponentState pour les requêtes de pages internes, est couverte dans ASP.NET Core Blazor routing and navigation.

Conseils de prérendu

Les conseils de prérendu sont organisés par sujets dans la documentation de Blazor. Les liens suivants couvrent tous les conseils de prérendu dans la documentation organisée par sujets :