成人怡红院-成人怡红院视频在线观看-成人影视大全-成人影院203nnxyz-美女毛片在线看-美女免费黄

站長資訊網
最全最豐富的資訊網站

一文探究Angular中的服務端渲染(SSR)

一文探究Angular中的服務端渲染(SSR)

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

本文是在 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
登錄后復制

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

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

  • 添加服務端文件:

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

    • 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)的網頁路徑,可以有幾種方式進行提供:

  • 通過命令行的附加參數:

    ng run <app-name>:prerender --routes /product/1 /product/2
    登錄后復制

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

    routes.txt

    /products/1 /products/23 /products/145 /products/555
    登錄后復制

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

    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"      ]    },
    登錄后復制

配置完成后,重新執行預渲染命令(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 技術。

贊(0)
分享到: 更多 (0)
?
網站地圖   滬ICP備18035694號-2    滬公網安備31011702889846號
肉身避风港1978大米星球| 色综合AV综合无码综合网站| 情人伊人久久综合亚洲| 日韩精品专区在线影院重磅| 四虎国产精品永久在线| 无码熟熟妇丰满人妻啪啪喷水 | 成人一区二区免费视频| 差差差软件大全APP推荐免费| 老熟妇午夜毛片一区二区三区| 国产SP调教打屁股视频网站| 国产精品久久婷婷六月丁香| 国产中国男男GayGay| 久久精品香蕉绿巨人登场| 女人与牛ZOZO| 久久棈精品久久久久久噜噜| 免费毛片45分钟| 人人妻人人澡人人爽人人到DVD | 国产成人午夜精品影院| 国产精品R级最新在线观看| 激情都市 校园 人妻 武侠| 巨大垂乳日本熟妇挤奶| 人妻一区二区三区Av毛片| 无码人妻AⅤ一区二区三区蜜桃| 少妇被躁爽到呻吟全过的小说| 人妻插B视频一区二区三区| 婷婷五月花丁香综合| 亚洲GV天堂GV无码男同在线观| 西西人体444WWW高清大胆| 亚洲男人第一无码AV网站| 69精产国品一二三产区区别| 岛国岛国免费V片在线观看| 国精产品一品二品国精HTC| 老男人把舌头伸进我下面| 日本人妻丰满大屁股a v| 亚洲AV成人无码深夜高潮| 又硬又粗又大一区二区三区视频| 荫蒂每天被三个男人添视频| 爱性久久久久久久久| 国产性大战XXXXX久久久| 看国产黄大片在线观看| 少妇短裙公车被直接进入| 亚洲六月丁香色婷婷综合久久| 亚洲中文字幕无码爆乳AV| 扒开两腿中间缝流白浆在线看| 锕锕锕锕锕锕~好湿WWW| 国产精品亚洲VA在线| 久久综合九色综合欧美| 肉妇春潮干柴烈火MYFDUCC| 亚洲精品无码AⅤ片桃花岛| S货叫大点声C烂你的SB视频| 国产制服丝袜在线无码| 欧美日韩免费观看| 亚洲WWW永久成人网站| ZO2O女人另类ZO2O洗浴| 激情五月综合 香亚洲| 欧美最猛黑人XXXXWWW| 亚洲AV午夜成人片精品网站| CAOPOREN个人免费公开| 韩国三级香港三级日本三级L| 久久精品国产亚洲AV网站| 色噜噜噜亚洲男人的天堂| 亚洲欧洲成人A∨在线观看| 超碰成人人人做人人爽| 久久久精品人妻人人澡| 丝瓜草莓视频APP| 最新系列国产专区|亚洲国产| 啊灬啊灬啊灬快灬高潮少妇| 好大好深好猛好爽视频| 日本无人区码卡二卡三卡| 野花高清在线观看免费3| 国产丰满饥渴老女人HD| 欧美A级情欲片在线观看免费 | 无码成人H动漫在线网站| 月光影院在线观看免费直播| 国产欧美久久一区二区| 人妻少妇看A偷人无码精品视频| 天天AV天天翘天天综合网色鬼| 亚洲AV成人无码一区在线观看| 亚洲女毛多水多21P| 成人无码免费一区二区三区| 久久久人人人婷婷色东京热| 无套内射GIF舔B吃奶| 斑马视频电影免费观看| 久久丫精品国产亚洲AV| 亚洲AV成人片无码网站网| 成人免费无遮挡在线播放| 美女图片禁欲系高级感| 亚洲第一综合天堂另类专| 国产69精品久久久久9999A| 女性高爱潮免费有声视频网站| 天堂いっしょにしよ在线| AA区一区二区三无码精片| 久久精品国产亚洲AV嫖农村妇女 | 日韩综合无码一区二区| 中国另类丰满熟妇乱XXXXX| 国产在线精品99一区不卡| 日韩人妻精品无码一区二区三区| 亚洲VA欧洲VA日韩VA| 东北一家人1一6全文阅读小说| 精品国产污污免费网站入口| 无码人妻一区二区三区麻豆| 成人毛片18女人毛片免费看快色| 果冻传媒亚洲区二期| 少妇系列之白嫩人妻| 边摸边吃奶又黄又激烈视频| 免费看高清毛片AAAAAAAA| 亚洲中文字幕无码中文 | 青青草国产成人A∨| 中文字幕乱码一区二区三区免费| 公和熄小婷乱中文字幕| 欧美人妻一区二区| 中文字幕无码成人片| 老女人性饥渴XXXXⅩHD| 亚洲色成人四虎在线观看| 狠狠色伊人亚洲综合网站野外 | 日韩精品无码专区免费播放| 2019四虎影视最新在线| 巨人精品福利官方导航| 亚洲一线产区二线产区区别在哪里| 疯狂做爰XXXⅩ高潮69短| 強暴強姦AV正片一区二区| 99热都是精品久久久久久| 男人把女人桶到爽免费应用 | 野花香电视剧全集免费观看| 皇叔撞着小公主的小说叫什么| 日本丶国产丶欧美色综合| 777久久精品一区二区三区无码| 国语偷拍人妻露脸| 翁熄小莹女博士高潮连连| 国产XXXX99真实实拍| 天天躁日日躁狠狠躁欧美老妇小说| 67194熟妇在线直接进入百度| 国内精品久久久久久中文字幕| 热の无码热の有码热の综合| 99国产精品欧美一区二区三区| 国内精品人妻无码久久久影院| 人人妻人人澡人人爽不卡视频| 亚洲欧洲中文日韩乱码AV| 精品熟人妻一区二区三区在线| 色一情一区二区三区四区| 成人国产一区二区三区精品不卡| 免费观看四虎精品国产地址| 中文在线中文资源| 女人天堂亚洲AⅤ在线观看| 97夜夜澡人人双人人人喊| 女学生14毛片视频片二毛| JAPANESEHD无码专区| 日本最新免费二区| 国产AⅤ无码专区亚洲AV| 天天天狠天天碰天天爱| 国产剧情AV在线| 亚洲AV涩涩涩成人网站在线播放| 国产成人AV片无码免费| 无码熟妇人妻AV在线影片最多| 波多野成人无码精品电影| 日韩在线观看视频一区二区| 国产成 人 黄 色 网 站 小说| 日本人妻和黑人又粗又长又黄| 中文无码久久精品| 欧洲肉欲K8播放毛片| 绯色AV一区二区三区在线高清| 欧美性BBBBBXXXXX4050免费看| 一区二区三区午夜无码视频| 蜜乳AV一区二区三区四区| STEAMWORKSHOP魅魔| 少妇伦子伦情品无吗| 国产亚洲欧美日韩亚洲中文色| 我和岳乱妇三级高清电影| 国产尤物精品视频| 一本久久精品一区二区| 欧美成人一区二区三区| 丰满人妻妇伦又伦精品国产| 亚洲 欧美 综合 在线 精品| 久久久国产精品ⅤA麻豆LL| CHINESE国产老熟女| 牲欲强的熟妇农村老妇女视频| 成人毛片18女人毛片免费看快色 | 中国熟妇色XXXXX中国老妇| 欧美巨大XXXX做受| 国产94在线 | 传媒麻豆| 亚洲AV中文无码乱人伦| 馒头型B好还是蝴蝶型B| 出轨 无码 论坛| 亚洲AV午夜成人片忘忧草在线 | 久久久WWW成人免费毛片| 99精品无人区乱码1区2区3区| 蜜乳AV一区二区三区| 成人A级毛片免费观看| 性姿势108式大全图解| 久久夜色精品国产| 成人免费777777被爆出| 亚洲AV无码一区二区三区波多野| 娇妻被朋友在客厅呻吟动漫| 18禁高潮出水呻吟娇喘蜜芽| 丝袜AV在线丝袜AV天堂| 久久精品人人槡人妻人人玩AV| 亚洲裸男GAY自慰网站| 欧美精品成人久久AV爱乃娜美 | 欧美疯狂性受XXXXX喷水| 国产AV无码专区影视|