一文探究Angular中的服務(wù)端渲染(SSR)-環(huán)球關(guān)注

          來源:php中文網(wǎng) | 2022-12-27 19:54:07 |

          一般來說,普通的 Angular 應(yīng)用是在 瀏覽器中運行,在 DOM 中對頁面進(jìn)行渲染,并與用戶進(jìn)行交互。而 Angular Universal 是在 服務(wù)端進(jìn)行渲染(Server-Side Rendering,SSR),生成靜態(tài)的應(yīng)用程序網(wǎng)頁,然后在客戶端展示,好處是可以更快地進(jìn)行渲染,在提供完整的交互之前就可以為用戶提供內(nèi)容展示。【相關(guān)教程推薦:《angular教程》】


          (資料圖片)

          使用 SSR 的好處

          對 SEO 更加友好

          雖然現(xiàn)在包括 Google 在內(nèi)的某些搜索引擎和社交媒體聲稱已經(jīng)能支持對由 JavaScript(JS)驅(qū)動的 SPA(Single-Page Application)應(yīng)用進(jìn)行爬取,但是結(jié)果似乎差強(qiáng)人意。靜態(tài) HTML 網(wǎng)站的 SEO 表現(xiàn)還是要好于動態(tài)網(wǎng)站,這也是 Angular 官網(wǎng)所持有的觀點(Angular 可是 Google 的!)。

          Universal 可以生成無 JS 的靜態(tài)版本的應(yīng)用程序,對搜索、外鏈、導(dǎo)航的支持更好。

          提高移動端的性能

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

          更快地展示首頁

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

          使用了 Universal 之后,應(yīng)用程序的首頁會以完整的形態(tài)展示給用戶,這是純的 HTML 網(wǎng)頁,即使不支持 JS,也可以展示。此時,網(wǎng)頁雖然不能處理瀏覽器的事件,但是支持通過 routerLink進(jìn)行跳轉(zhuǎn)。

          這么做的好處是,我們可以先用靜態(tài)網(wǎng)頁抓住用戶的注意力,在用戶瀏覽網(wǎng)頁的時候,同時加載整個 Angular 應(yīng)用。這給了用戶一個非常好的極速加載的體驗。

          為項目增加 SSR

          Angular CLI 可以幫助我們非常便捷的將一個普通的 Angular 項目轉(zhuǎn)變?yōu)橐粋€帶有 SSR 的項目。創(chuàng)建服務(wù)端應(yīng)用只需要一個命令:

          ng add @nguniversal/express-engine

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

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

          添加服務(wù)端文件:

          main.server.ts- 服務(wù)端主程序文件app/app.server.module.ts- 服務(wù)端應(yīng)用程序主模塊tsconfig.server.json- TypeScript 服務(wù)端配置文件server.ts- Express web server 的運行文件

          修改的文件:

          package.json- 添加 SSR 所需要的依賴和運行腳本angular.json- 添加開發(fā)、構(gòu)建 SSR 應(yīng)用所需要的配置

          package.json中,會自動添加一些 npm 腳本:dev:ssr用于在開發(fā)環(huán)境運行 SSR 版本;serve:ssr用于直接運行 build 或 prerender 后的網(wǎng)頁;build:ssr構(gòu)建 SSR 版本的網(wǎng)頁;prerender構(gòu)建預(yù)渲染后的網(wǎng)頁,與 build不同,這里會根據(jù)提供的 routes生成這些頁面的 HTML 文件。

          替換瀏覽器 API

          由于 Universal 應(yīng)用不是在瀏覽器中執(zhí)行,因此一些瀏覽器的 API 或功能將不可用。例如,服務(wù)端應(yīng)用是無法使用瀏覽器中的全局對象 windowdocumentnavigatorlocation

          Angular 提供了兩個可注入對象,用于在服務(wù)端替換對等的對象:LocationDOCUMENT

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

          import { Location } from "@angular/common"; export class AbmNavbarComponent implements OnInit{  // ctor 中注入 Location  constructor(private _location:Location){    //...  }   ngOnInit() {    // 打印當(dāng)前地址    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 應(yīng)用中,HTTP 請求的 URL 地址必須為 絕對地址(即,以 http/https開頭的地址,不能是相對地址,如 /api/heros)。Angular 官方推薦將請求的 URL 全路徑設(shè)置到 renderModule()renderModuleFactory()options參數(shù)中。但是在 v14 自動生成的代碼中,并沒有顯式調(diào)用這兩個方法的代碼。而通過讀 Http 請求的攔截,也可以達(dá)到同樣的效果。

          下面我們先準(zhǔn)備一個攔截器,假設(shè)文件位于項目的 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, 相對協(xié)議地址const isAbsoluteURL = startsWithAny(["http", "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 中獲取協(xié)議和地址                protocolHost = `${this.request.protocol}://${this.request.get(                    "host"                )}`;            } else {                // 如果注入的 REQUEST 為空,比如在進(jìn)行 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,這里也需要引入服務(wù)端模塊    FlexLayoutServerModule,   ],  providers: [    {      provide: HTTP_INTERCEPTORS,      useClass: UniversalRelativeInterceptor,      multi: true    }  ],  bootstrap: [AppComponent],})export class AppServerModule { }

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

          Prerender 預(yù)渲染靜態(tài) HTML

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

          prerendernpm run prerender)會在構(gòu)建時生成靜態(tài) HTML 文件。比如我們做企業(yè)官網(wǎng),只有幾個頁面,那么我們可以使用預(yù)渲染技術(shù)生成這幾個頁面的靜態(tài) HTML 文件,避免在運行時動態(tài)生成,從而進(jìn)一步提升網(wǎng)頁的訪問速度和用戶體驗。

          預(yù)渲染路徑配置

          需要進(jìn)行預(yù)渲染(預(yù)編譯 HTML)的網(wǎng)頁路徑,可以有幾種方式進(jìn)行提供:

          通過命令行的附加參數(shù):

          ng run <app-name>:prerender --routes /product/1 /product/2

          如果路徑比較多,比如針對 product/:id這種動態(tài)路徑,則可以使用一個路徑文件:

          routes.txt

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

          然后在命令行參數(shù)指定該文件:

          ng run <app-name>:prerender --routes-file routes.txt

          在項目的 angular.json文件配置需要的路徑:

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

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

          SEO 優(yōu)化

          SEO 的關(guān)鍵在于對網(wǎng)頁 titlekeywordsdescription的收錄,因此對于我們想要讓搜索引擎收錄的網(wǎng)頁,可以修改代碼提供這些內(nèi)容。

          在 Angular 14 中,如果路由界面通過 Routes配置,可以將網(wǎng)頁的靜態(tài) title直接寫在路由的配置中:

          { path: "home", component: AbmHomeComponent, title: "<你想顯示在瀏覽器 tab 上的標(biāo)題>" },

          另外,Angular 也提供了可注入的 TitleMeta用于修改網(wǎng)頁的標(biāo)題和 meta 信息:

          import { Meta, Title } from "@angular/platform-browser"; export class AbmHomeComponent implements OnInit {   constructor(    private _title: Title,    private _meta: Meta,  ) { }   ngOnInit() {    this._title.setTitle("<此頁的標(biāo)題>");    this._meta.addTags([      { name: "keywords", content: "<此頁的 keywords,以英文逗號隔開>" },      { name: "description", content: "<此頁的描述>" }    ]);  }}

          總結(jié)

          Angular 作為 SPA 企業(yè)級開發(fā)框架,在模塊化、團(tuán)隊合作開發(fā)方面有自己獨到的優(yōu)勢。在進(jìn)化到 v14 這個版本中提供了不依賴 NgModule的獨立 Component功能,進(jìn)一步簡化了模塊化的架構(gòu)。

          Angular Universal 主要關(guān)注將 Angular App 如何進(jìn)行服務(wù)端渲染和生成靜態(tài) HTML,對于用戶交互復(fù)雜的 SPA 并不推薦使用 SSR。針對頁面數(shù)量較少、又有 SEO 需求的網(wǎng)站或系統(tǒng),則可以考慮使用 Universal 和 SSR 技術(shù)。

          更多編程相關(guān)知識,請訪問:編程教學(xué)!!

          以上就是一文探究Angular中的服務(wù)端渲染(SSR)的詳細(xì)內(nèi)容,更多請關(guān)注php中文網(wǎng)其它相關(guān)文章!

          關(guān)鍵詞: Angular

          亚洲爱情岛论坛永久| 亚洲国产综合第一精品小说| 亚洲午夜国产精品无码老牛影视| 亚洲中文字幕AV在天堂| 亚洲a在线视频视频| 亚洲人色婷婷成人网站在线观看| 亚洲人成电影网站国产精品| 国产精品亚洲综合天堂夜夜| 亚洲AV无码专区亚洲AV桃| 亚洲色成人网站WWW永久四虎| 2020国产精品亚洲综合网 | 亚洲白色白色永久观看| 久久久久亚洲av无码专区喷水| 亚洲AV无码成人专区片在线观看| 亚洲αv在线精品糸列| 亚洲精品乱码久久久久久按摩 | 亚洲自偷精品视频自拍| 亚洲AV综合色一区二区三区| 国产综合精品久久亚洲| 亚洲第一页日韩专区| 亚洲无码一区二区三区| 亚洲永久在线观看| 亚洲中文无码线在线观看| 亚洲熟妇av一区二区三区下载| 香蕉视频在线观看亚洲| 亚洲成a人片在线观看久| 亚洲精品乱码久久久久蜜桃 | 亚洲色成人四虎在线观看| 亚洲AV成人影视在线观看| 亚洲一级毛片在线播放| 亚洲日本乱码卡2卡3卡新区| 亚洲精品电影在线| 久久国产亚洲高清观看| 色婷婷亚洲十月十月色天| 中文字幕不卡亚洲| 最新国产AV无码专区亚洲| 亚洲一区AV无码少妇电影☆| 亚洲女同成av人片在线观看| 人人狠狠综合久久亚洲88| 亚洲精品无码高潮喷水在线| 亚洲av午夜成人片精品网站|