共用方式為


預先轉譯 ASP.NET Core Razor 元件

注意

這不是這篇文章的最新版本。 如需目前的版本,請參閱 本文的 .NET 9 版本。

重要

這些發行前產品的相關資訊在產品正式發行前可能會有大幅修改。 Microsoft 對此處提供的資訊,不做任何明確或隱含的瑕疵擔保。

如需目前的版本,請參閱 本文的 .NET 9 版本。

本文說明 Razor中伺服器轉譯元件的 Blazor Web App 元件預先轉譯案例。

預先轉譯是伺服器上初始轉譯頁面內容,而不需要為轉譯的控制項啟用事件處理常式的程序。 伺服器會盡快輸出頁面的 HTML UI,以回應初始要求,如此可改善應用程式對使用者的回應速度。 預先轉譯也可改善搜尋引擎最佳化 (SEO),方法是轉譯搜尋引擎用來計算頁面排名的初始 HTTP 回應的內容。

保存預先轉譯的狀態

若未保存預先轉譯的狀態,在預先轉譯期間所使用的狀態會遺失,而且必須在應用程式完全載入時重建。 如果以非同步方式建立任何狀態,UI 可能會閃爍,因為在元件重新轉譯時會取代預先轉譯的 UI。

請考慮下列 PrerenderedCounter1 計數器元件。 元件在 OnInitialized 生命週期方法中預先轉譯期間設定初始隨機計數器值。 建立與用戶端的 SignalR 連線之後,元件會重新轉譯,並在第二次執行 OnInitialized 時取代初始計數值。

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++;
}

執行應用程式並檢查來自元件的記錄: 以下是範例輸出。

注意

如果應用程式採用 互動式路由,且頁面是透過內部 增強導覽到達,則不會發生預先呈現。 因此,您必須對 PrerenderedCounter1 元件執行完整頁面重新載入,才能看到下列輸出。 如需詳細資訊,請參閱 互動式路由和預先呈現 一節。

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

第一個記錄的計數會在預先轉譯期間發生。 重新轉譯元件後,會再次設定此計數。 當計數從 41 更新為 92 時,UI 中也會閃爍。

若要在預先轉譯期間保留計數器的初始值,Blazor 會使用 PersistentComponentState 服務支援在預先轉譯的頁面中保存狀態 (以及針對內嵌至 Razor Pages 或 MVC 應用程式的頁面或檢視中的元件,使用 保存元件狀態標記協助程式)。

若要保留預先轉譯的狀態,請決定要使用 PersistentComponentState 服務保存哪個狀態。 PersistentComponentState.RegisterOnPersisting 會註冊回呼,以在應用程式暫停前保存元件狀態。 應用程式繼續時會擷取此狀態。

下列範例示範一般模式:

  • {TYPE} 預留位置代表要保存的資料類型。
  • {TOKEN} 預留位置是狀態識別碼字串。 請考慮使用 nameof({VARIABLE}),其中 {VARIABLE} 預留位置是保存狀態的變數名稱。 針對狀態識別碼使用 nameof() 可避免使用引號字串。
@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();
    }
}

下列計數器元件範例會在預先轉譯期間保存計數器狀態,並擷取要初始化元件的狀態。

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++;
}

當元件執行時,currentCount 只會在預先轉譯期間設定一次。 重新轉譯元件時,會還原此值。 以下是範例輸出。

注意

如果應用程式採用 互動式路由,且頁面是透過內部 增強導覽到達,則不會執行預先渲染。 因此,您必須對 PrerenderedCounter2 元件執行完整頁面重新載入,才能看到下列輸出。 如需詳細資訊,請參閱 互動式路由和預先呈現 一節。

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

在預先轉譯期間使用相同的狀態初始化元件,任何昂貴的初始化步驟只會執行一次。 轉譯的 UI 也會符合預先轉譯的 UI,因此瀏覽器中不會發生閃爍。

保存的預先呈現狀態會傳送至用戶端,用來還原元件狀態。 在用戶端轉譯 (CSR, ) 期間, InteractiveWebAssembly數據會公開給瀏覽器,且不得包含機密的私人資訊。 在互動式伺服器端轉譯期間(互動式 SSR, InteractiveServer), ASP.NET Core Data Protection 可確保數據安全地傳輸。 轉 InteractiveAuto 譯模式結合了 WebAssembly 和伺服器互動功能,因此必須考慮數據暴露在瀏覽器上,如同 CSR 案例一樣。

嵌入至頁面和檢視的元件(Razor Pages/MVC)

對於內嵌至 Razor Pages 和 MVC 應用程式的頁面或檢視中的元件,您必須在應用程式版面配置的結尾 標記內新增具有 <persist-component-state /> HTML 標記的</body>只有 Razor Pages 和 MVC 應用程式才需要此協助程式。 如需詳細資訊,請參閱 ASP.NET Core 中的保存元件狀態標記協助程式

Pages/Shared/_Layout.cshtml

<body>
    ...

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

互動式路由和預先轉譯

Routes 元件未定義渲染模式時,應用程式會使用逐頁/元件互動和導航。 使用每頁/元件導航時,內部導航在應用程式變得互動式之後,由增強式路由來處理。 †在此上下文中,內部 指的是導航事件的 URL 目標位於應用程式中的 Blazor 端點。

PersistentComponentState 服務只能在初始頁面載入上運作,而不適用於內部增強式頁面流覽事件。

如果應用程式執行完整(非增強的)瀏覽至使用永續性元件狀態的頁面,則當應用程式變成互動式時,會提供持續性狀態。

如果已建立互動式線路,且增強式導覽會執行至使用永續性元件狀態的頁面,則現有線路中不會提供狀態 ,讓元件使用。 內部頁面要求沒有預渲染,且 PersistentComponentState 服務不知道已發生增強式導覽。 沒有機制可傳遞狀態更新至已在現有線路上執行的元件。 原因是 Blazor 只支援在運行時間初始化時將狀態從伺服器傳遞至用戶端,而不是在運行時間啟動之後。

正在考慮在 .NET 10(2025 年 11 月)中對 Blazor 架構進行額外工作,以解決此情況。 如需 不支援因應措施的詳細資訊和社群討論,請參閱 支援增強頁面導覽 (dotnet/aspnetcore #51584)的持續元件狀態。 •不支援的因應措施不受Microsoft批准,以用於 Blazor 應用程式。 使用第三方套件、方法和程序代碼,並自行承擔風險。

停用增強式流覽涵蓋在 ASP.NET Core Blazor 路由和流覽中,這雖然會降低效能,但也能避免載入內部頁面要求時遇到狀態 PersistentComponentState 的問題。

預先轉譯指引

預先轉譯指引會依主題在 Blazor 文件中組織。 下列連結涵蓋依主題設定的文件內的所有預先轉譯指引: