久久精品五月,日韩不卡视频在线观看,国产精品videossex久久发布 ,久久av综合

站長資訊網(wǎng)
最全最豐富的資訊網(wǎng)站

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

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

一般來說,普通的 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教程》】

本文是在 Angular 14 環(huán)境中完成,有些內(nèi)容對于新的 Angular 版本可能并不適用,請參考 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
登錄后復(fù)制

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

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

  • 添加服務(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));   } }
登錄后復(fù)制

同樣,對于在瀏覽器使用 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');   } }
登錄后復(fù)制

使用 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', '//']);   @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);         }     } }
登錄后復(fù)制

然后在 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 { }
登錄后復(fù)制

這樣任何對于相對地址的請求都會自動轉(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
    登錄后復(fù)制

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

    routes.txt

    /products/1 /products/23 /products/145 /products/555
    登錄后復(fù)制

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

    ng run <app-name>:prerender --routes-file routes.txt
    登錄后復(fù)制

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

     "prerender": {    "builder": "@nguniversal/builders:prerender",    "options": {      "routes": [ // 這里配置        "/",        "/main/home",        "/main/service",        "/main/team",        "/main/contact"      ]    },
    登錄后復(fù)制

配置完成后,重新執(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)題>' },
登錄后復(fù)制

另外,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: '<此頁的描述>' }     ]);   } }
登錄后復(fù)制

總結(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ù)。

贊(0)
分享到: 更多 (0)
?
網(wǎng)站地圖   滬ICP備18035694號-2    滬公網(wǎng)安備31011702889846號
久久精品五月,日韩不卡视频在线观看,国产精品videossex久久发布 ,久久av综合
激情久久久久久| 亚洲免费在线| 国产自产自拍视频在线观看| 久久影院资源站| 亚洲天堂资源| 尤物在线精品| 蜜臀av在线播放一区二区三区| 综合色一区二区| 国产精品探花在线观看| 97精品一区| 久久亚洲风情| 美女久久久久久| 亚洲欧美日韩高清在线| 91精品国产自产观看在线| 超碰成人av| 石原莉奈一区二区三区在线观看| 欧美成a人片免费观看久久五月天| 日韩欧美在线中字| 亚洲免费毛片| 日韩欧美另类一区二区| 在线观看亚洲精品福利片| 久久超碰99| 国精品一区二区三区| 日韩av中文在线观看| 日韩欧美三级| 欧美亚洲一区二区三区| 999久久久国产精品| 日本欧美在线| 婷婷亚洲五月| 久久97视频| 亚洲+小说+欧美+激情+另类| 亚洲欧洲高清| 国产精品午夜一区二区三区| 在线视频免费在线观看一区二区| 久久免费福利| 免费观看在线综合| 国产日韩电影| 91大神在线观看线路一区| 在线日韩一区| 欧美a级一区二区| 综合激情网站| 亚洲欧美综合| 国产不卡精品| 日韩二区在线观看| 欧美天堂亚洲电影院在线观看| 麻豆中文一区二区| 日韩欧美激情| 国产精品毛片在线| 精品免费av在线| 国产精品17p| 亚洲三级视频| 欧美女激情福利| 欧美丰满日韩| 日本免费新一区视频| 免播放器亚洲| 免费国产自久久久久三四区久久 | 日韩在线精品| 国产精品调教视频| 日韩中文字幕| 久久国产精品久久久久久电车| 日韩免费久久| 国产一区二区精品久| 欧美一级网址| 亚洲精品伊人| 精品91久久久久| 成人羞羞在线观看网站| 精品国产亚洲日本| 丝袜国产日韩另类美女| 午夜免费一区| 欧洲毛片在线视频免费观看| 日韩免费在线| 国产粉嫩在线观看| 日韩成人a**站| 宅男噜噜噜66国产日韩在线观看| 久久久9色精品国产一区二区三区| 97在线精品| 久久精品国产999大香线蕉| 欧美精品国产一区| 日韩综合小视频| 亚洲精品麻豆| 久久午夜影视| 日本大胆欧美人术艺术动态| 99香蕉国产精品偷在线观看| 激情综合网址| 91精品电影| 日韩精品网站| 久久亚洲国产| 91久久久精品国产| 9国产精品视频| 9久re热视频在线精品| 99热精品在线观看| 午夜一级久久| 蜜桃av一区二区三区电影| 亚洲一区激情| av不卡免费看| 先锋亚洲精品| 日韩制服丝袜av| 亚洲字幕久久| 日韩av电影一区| 国产精品亲子伦av一区二区三区| 国产伦精品一区二区三区在线播放| 欧美日韩18| 美女性感视频久久| 国际精品欧美精品| 色网在线免费观看| 久久精选视频| 夜久久久久久| 日韩精品免费视频一区二区三区| 日韩国产欧美一区二区三区| 日韩欧美中文字幕一区二区三区| 欧美欧美黄在线二区| 国产精品66| 日韩av福利| 亚洲视频www| 日韩高清三区| 精品国产中文字幕第一页 | 天堂√8在线中文| 91精品啪在线观看国产18| 欧美日韩国产欧| 亚洲三级观看| 免费在线亚洲欧美| 亚洲日本网址| 午夜一级久久| 日韩高清国产一区在线| 久久伊人久久| 亚洲国产影院| 午夜精品影视国产一区在线麻豆| 欧美在线日韩| 久久精品理论片| 欧美~级网站不卡| 日韩精品国产欧美| 福利一区和二区| 99国产精品视频免费观看一公开| 日本午夜免费一区二区| 国产成人精品一区二区免费看京| 伊人久久大香线蕉av不卡| 亚洲久久视频| 精品一区二区三区中文字幕| 成人啊v在线| 日韩精品一二区| 精品国产a一区二区三区v免费| 国产综合视频| 久久国产精品色av免费看| 亚洲伊人av| 综合一区在线| 女生影院久久| 亚洲精品在线二区| 狠狠躁少妇一区二区三区| 亚洲欧美视频一区二区三区| 欧美精品1区| 最新日韩欧美| 欧美激情亚洲| 久久午夜精品| 激情不卡一区二区三区视频在线| 国产亚洲激情| 精品一区二区男人吃奶 | 丝袜亚洲另类欧美| 国产精品s色| 91成人网在线观看| 国产精品久久久久久av公交车| 欧美肉体xxxx裸体137大胆| 日本不卡一区二区三区| 日韩欧美网址| 国产乱码精品一区二区三区亚洲人| 久久国产免费| 国产精品一二| 久久午夜精品| 久久久久久自在自线| 涩涩涩久久久成人精品| 女生影院久久| 国产精品综合| 中文亚洲免费| 中文在线а√天堂 | 国产日产精品_国产精品毛片| 日韩精品欧美| 久久99青青| 日韩精品一级二级| 日本不良网站在线观看| 日韩精品第一| av亚洲在线观看| 岛国av在线播放| 91精品国产自产精品男人的天堂 | 国产精品hd| 亚洲欧洲美洲国产香蕉| 亚洲a一区二区三区| 久久久久久久久成人| 亚洲va久久| 国产精品日本欧美一区二区三区| 精品国产午夜肉伦伦影院| 亚洲精品在线国产| 欧美在线亚洲| 日韩毛片视频| 精品久久免费| 国产亚洲一区| 日韩一区欧美二区| 亚洲成人三区| 美女毛片一区二区三区四区| 麻豆视频在线观看免费网站黄| 国产精品自在| 91久久精品无嫩草影院|