Angular SSR 探究

来源:https://www.cnblogs.com/abmcode/archive/2022/11/04/angular_ssr_seo.html
-Advertisement-
Play Games

一般來說,普通的 Angular 應用是在 瀏覽器 中運行,在 DOM 中對頁面進行渲染,並與用戶進行交互。而 Angular Universal 是在 服務端 進行渲染(Server-Side Rendering,SSR),生成靜態的應用程式網頁,然後在客戶端展示,好處是可以更快地進行渲染,在提供 ...


一般來說,普通的 Angular 應用是在 瀏覽器 中運行,在 DOM 中對頁面進行渲染,並與用戶進行交互。而 Angular Universal 是在 服務端 進行渲染(Server-Side Rendering,SSR),生成靜態的應用程式網頁,然後在客戶端展示,好處是可以更快地進行渲染,在提供完整的交互之前就可以為用戶提供內容展示。

本文是在 Angular 14 環境中完成,有些內容對於新的 Angular 版本可能並不適用,請參考 Angular 官方文檔。

使用 SSR 的好處

對 SEO 更加友好

雖然現在包括 Google 在內的某些搜索引擎和社交媒體聲稱已經能支持對由 JavaScript(JS)驅動的 SPA(Single-Page Application)應用進行爬取,但是結果似乎差強人意。靜態 HTML 網站的 SEO 表現還是要好於動態網站,這也是 Angular 官網所持有的觀點(Angular 可是 Google 的!)。

Universal 可以生成無 JS 的靜態版本的應用程式,對搜索、外鏈、導航的支持更好。

提高移動端的性能

某些移動端設備可能不支持 JS 或者對 JS 的支持非常有限,導致網站的訪問體驗非常差。這種情況下,我們需要提供無 JS 版本的應用,以便為用戶提供更好的體驗。

更快地展示首頁

對於用戶的使用體驗來說,首頁展示速度的快慢至關重要。根據 eBay 的數據,搜索結果的展示速度每提高 100 毫秒,“添加至購物車”的使用率就提高 0.5%。

使用了 Universal 之後,應用程式的首頁會以完整的形態展示給用戶,這是純的 HTML 網頁,即使不支持 JS,也可以展示。此時,網頁雖然不能處理瀏覽器的事件,但是支持通過 routerLink 進行跳轉。

這麼做的好處是,我們可以先用靜態網頁抓住用戶的註意力,在用戶瀏覽網頁的時候,同時載入整個 Angular 應用。這給了用戶一個非常好的極速載入的體驗。

為項目增加 SSR

Angular CLI 可以幫助我們非常便捷的將一個普通的 Angular 項目轉變為一個帶有 SSR 的項目。創建服務端應用只需要一個命令:

ng add @nguniversal/express-engine

建議在運行該命令之前先提交所有的改動。

這個命令會對項目做如下修改:

  1. 添加服務端文件:

    • main.server.ts - 服務端主程式文件
    • app/app.server.module.ts - 服務端應用程式主模塊
    • tsconfig.server.json - TypeScript 服務端配置文件
    • server.ts - Express web server 的運行文件
  2. 修改的文件:

    • package.json - 添加 SSR 所需要的依賴和運行腳本
    • angular.json - 添加開發、構建 SSR 應用所需要的配置

package.json 中,會自動添加一些 npm 腳本:dev:ssr 用於在開發環境運行 SSR 版本;serve:ssr 用於直接運行 build 或 prerender 後的網頁;build:ssr 構建 SSR 版本的網頁;prerender 構建預渲染後的網頁,與 build 不同,這裡會根據提供的 routes 生成這些頁面的 HTML 文件。

替換瀏覽器 API

由於 Universal 應用不是在瀏覽器中執行,因此一些瀏覽器的 API 或功能將不可用。例如,服務端應用是無法使用瀏覽器中的全局對象 windowdocumentnavigatorlocation

Angular 提供了兩個可註入對象,用於在服務端替換對等的對象:LocationDOCUMENT

例如,在瀏覽器中,我們通過 window.location.href 獲取當前瀏覽器的地址,而改成 SSR 之後,代碼如下:

import { Location } from '@angular/common';

export class AbmNavbarComponent implements OnInit{
  // ctor 中註入 Location
  constructor(private _location:Location){
    //...
  }

  ngOnInit() {
    // 列印當前地址
    console.log(this._location.path(true));
  }
}

同樣,對於在瀏覽器使用 document.getElementById() 獲取 DOM 元素,在改成 SSR 之後,代碼如下:

import { DOCUMENT } from '@angular/common';

export class AbmFoxComponent implements OnInit{
  // ctor 中註入 DOCUMENT
  constructor(@Inject(DOCUMENT) private _document: Document) { }

  ngOnInit() {
    // 獲取 id 為 fox-container 的 DOM
    const container = this._document.getElementById('fox-container');
  }
}

使用 URL 絕對地址

在 Angular SSR 應用中,HTTP 請求的 URL 地址必須為 絕對地址(即,以 http/https 開頭的地址,不能是相對地址,如 /api/heros)。Angular 官方推薦將請求的 URL 全路徑設置到 renderModule()renderModuleFactory()options 參數中。但是在 v14 自動生成的代碼中,並沒有顯式調用這兩個方法的代碼。而通過讀 Http 請求的攔截,也可以達到同樣的效果。

下麵我們先準備一個攔截器,假設文件位於項目的 shared/universal-relative.interceptor.ts 路徑:

import { HttpHandler, HttpInterceptor, HttpRequest } from '@angular/common/http';
import { Inject, Injectable, Optional } from '@angular/core';
import { REQUEST } from '@nguniversal/express-engine/tokens';
import { Request } from 'express';

// 忽略大小寫檢查
const startsWithAny = (arr: string[] = []) => (value = '') => {
    return arr.some(test => value.toLowerCase().startsWith(test.toLowerCase()));
};

// http, https, 相對協議地址
const isAbsoluteURL = startsWithAny(['http', '//']);

@Injectable()
export class UniversalRelativeInterceptor implements HttpInterceptor {
    constructor(@Optional() @Inject(REQUEST) protected request: Request) { }

    intercept(req: HttpRequest<any>, next: HttpHandler) {
        // 不是絕對地址的 URL
        if (!isAbsoluteURL(req.url)) {
            let protocolHost: string;
            if (this.request) {
                // 如果註入的 REQUEST 不為空,則從註入的 SSR REQUEST 中獲取協議和地址
                protocolHost = `${this.request.protocol}://${this.request.get(
                    'host'
                )}`;
            } else {
                // 如果註入的 REQUEST 為空,比如在進行 prerender build:
                // 這裡需要添加自定義的地址首碼,比如我們的請求都是從 abmcode.com 來。
                protocolHost = 'https://www.abmcode.com';
            }
            const pathSeparator = !req.url.startsWith('/') ? '/' : '';
            const url = protocolHost + pathSeparator + req.url;
            const serverRequest = req.clone({ url });
            return next.handle(serverRequest);

        } else {
            return next.handle(req);
        }
    }
}

然後在 app.server.module.ts 文件中 provide 出來:

import { UniversalRelativeInterceptor } from './shared/universal-relative.interceptor';
// ... 其他 imports

@NgModule({
  imports: [
    AppModule,
    ServerModule,
    // 如果你用了 @angular/flext-layout,這裡也需要引入服務端模塊
    FlexLayoutServerModule, 
  ],
  providers: [
    {
      provide: HTTP_INTERCEPTORS,
      useClass: UniversalRelativeInterceptor,
      multi: true
    }
  ],
  bootstrap: [AppComponent],
})
export class AppServerModule { }

這樣任何對於相對地址的請求都會自動轉換為絕對地址請求,在 SSR 的場景下不會再出問題。

Prerender 預渲染靜態 HTML

經過上面的步驟後,如果我們通過 npm run build:ssr 構建項目,你會發現在 dist/<your project>/browser 下麵只有 index.html 文件,打開文件查看,發現其中還有 <app-root></app-root> 這樣的元素,也就是說你的網頁內容並沒有在 html 中生成。這是因為 Angular 使用了動態路由,比如 /product/:id 這種路由,而頁面的渲染結果要經過 JS 的執行才能知道,因此,Angular 使用了 Express 作為 Web 伺服器,能在服務端運行時根據用戶請求(爬蟲請求)使用模板引擎生成靜態 HTML 界面。

prerendernpm run prerender)會在構建時生成靜態 HTML 文件。比如我們做企業官網,只有幾個頁面,那麼我們可以使用預渲染技術生成這幾個頁面的靜態 HTML 文件,避免在運行時動態生成,從而進一步提升網頁的訪問速度和用戶體驗。

預渲染路徑配置

需要進行預渲染(預編譯 HTML)的網頁路徑,可以有幾種方式進行提供:

  1. 通過命令行的附加參數:

    ng run <app-name>:prerender --routes /product/1 /product/2
    
  2. 如果路徑比較多,比如針對 product/:id 這種動態路徑,則可以使用一個路徑文件:

    routes.txt

    /products/1
    /products/23
    /products/145
    /products/555
    

    然後在命令行參數指定該文件:

    ng run <app-name>:prerender --routes-file routes.txt
    
  3. 在項目的 angular.json 文件配置需要的路徑:

     "prerender": {
       "builder": "@nguniversal/builders:prerender",
       "options": {
         "routes": [ // 這裡配置
           "/",
           "/main/home",
           "/main/service",
           "/main/team",
           "/main/contact"
         ]
       },
    

配置完成後,重新執行預渲染命令(npm run prerender 或者使用命令行參數則按照上面<1><2>中的命令執行),編譯完成後,再打開 dist/<your project>/browser 下的 index.html 會發現裡面沒有 <app-root></app-root> 了,取而代之的是主頁的實際內容。同時也生成了相應的路徑目錄以及各個目錄下的 index.html 子頁面文件。

SEO 優化

SEO 的關鍵在於對網頁 titlekeywordsdescription 的收錄,因此對於我們想要讓搜索引擎收錄的網頁,可以修改代碼提供這些內容。

在 Angular 14 中,如果路由界面通過 Routes 配置,可以將網頁的靜態 title 直接寫在路由的配置中:

{ path: 'home', component: AbmHomeComponent, title: '<你想顯示在瀏覽器 tab 上的標題>' },

另外,Angular 也提供了可註入的 TitleMeta 用於修改網頁的標題和 meta 信息:

import { Meta, Title } from '@angular/platform-browser';

export class AbmHomeComponent implements OnInit {

  constructor(
    private _title: Title,
    private _meta: Meta,
  ) { }

  ngOnInit() {
    this._title.setTitle('<此頁的標題>');
    this._meta.addTags([
      { name: 'keywords', content: '<此頁的 keywords,以英文逗號隔開>' },
      { name: 'description', content: '<此頁的描述>' }
    ]);
  }
}

總結

Angular 作為 SPA 企業級開發框架,在模塊化、團隊合作開發方面有自己獨到的優勢。在進化到 v14 這個版本中提供了不依賴 NgModule 的獨立 Component 功能,進一步簡化了模塊化的架構。

Angular Universal 主要關註將 Angular App 如何進行服務端渲染和生成靜態 HTML,對於用戶交互複雜的 SPA 並不推薦使用 SSR。針對頁面數量較少、又有 SEO 需求的網站或系統,則可以考慮使用 Universal 和 SSR 技術。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 摘要:GaussDB(for Influx)是一款基於計算存儲分離架構,完全相容 InfluxDB 生態的雲原生時序資料庫。 本文分享自華為雲社區《雲資料庫 GaussDB(for Influx) 解密第十一期:讓智能電網中時序數據處理更高效》,作者:華為雲資料庫 GaussDB(for Influ ...
  • CitusData於日前推出了Citus11.0,並宣佈將所有企業版的特性都進行開源。此前,Citus在版本更新時通常會同步發佈2個版本:開源版本和包含一些額外功能的企業版本。“但是,Citus11.0將只有一個版本,因為Citus擴展中的所有內容現在都是完全開源的!” 公告指出,這意味著你現在可... ...
  • 一、引言 理解和分析這個數據包結構(這裡面也涉及廣播間隔時間的設置,設備廣播數據間隔設置長了,會影響設備被髮現的效率;設置短時,又響應功耗)。 我們所說的BLE設備,其實是有區分有兩種角色 Central 和 Peripheral,也就是中心設備和外圍設備。中心設備可以主動連接外圍設備,外圍設備發送 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 相信用過vue的小伙伴,肯定被面試官問過這樣一個問題:在vue中動態的引入圖片為什麼要使用require 有些小伙伴,可能會輕蔑一笑:呵,就這,因為動態添加src被當做靜態資源處理了,沒有進行編譯,所以要加上require, 我倒著都能背 ...
  • 有這樣一道面試題:nextTick是什麼? 我們做如下實驗,在磁碟任意的位置(確保今後可以想起來),新建nextTick文件夾(可以命名為其他的)。 通過命令vue create demo創建以demo命名的vue2項目。 為了方便調試項目,我們通過VS Code打開創建的vue2項目demo。 我 ...
  • 一,面試基礎 HTML和CSS ps:這倆面試答不上來的,基本就可以回去了,以下是HTML題,一般來說這地方不會出太多題,面試官也不願意花太多時間在這上面。 1,HTML語義化,如何理解語義化? 讓人更容易懂(增加代碼的可讀性) 讓搜索引擎更容易懂,有利於爬蟲抓取 在沒有css的情況下,頁面也能更好 ...
  • 本文開始,首先我們來看這兩個詞的意思,provide:提供 inject:註入 用處: 父組件可以向其所有子組件傳入數據,而“不管子組件層次結構有多深(非父子和父子咱都能傳)” 特性: 父組件有一個provide選項來提供數據 子組件有一個inject選項來開始使用這個數據 本文參考組件層級: In ...
  • 1. 撲朔迷離的 this 從錶面來看,this 總能通過各種變通的方式得到意想不到的結果 既然是“意想不到”,就說明對“各種變通的方式”不太瞭解 那麼我們來看看“各種變通的方式” 老早以前,this 指向它的調用者 1 <script> 2 // 通過 dom 調用 3 var _html = d ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...