国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 編程 > JavaScript > 正文

詳解基于Angular4+ server render(服務(wù)端渲染)開(kāi)發(fā)教程

2019-11-19 15:39:10
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

目標(biāo):

1.更好的 SEO,方便搜索爬蟲(chóng)抓取頁(yè)面內(nèi)容

2.更快的內(nèi)容到達(dá)時(shí)間(time-to-content)

影響:

1.用戶:比原來(lái)更快的看到渲染的頁(yè)面,提升用戶體驗(yàn)

2.開(kāi)發(fā)人員:某些代碼可能需要特殊處理,才能在服務(wù)器渲染應(yīng)用程序中運(yùn)行(window,document, navigator等)

安裝:

1.nodejs 建議6+

2.angular建議4.1+

理論實(shí)現(xiàn):

盡管這是一張來(lái)自vue官網(wǎng)服務(wù)器渲染的一張示意圖,但是原理上和angular都是一樣的,只是實(shí)現(xiàn)的代碼不一致。

SSR 有兩個(gè)入口文件,app-client.js 和 app-server.js, 都包含了應(yīng)用代碼(appmodule),webpack 通過(guò)兩個(gè)入口文件分別打包成給服務(wù)端用的 server bundle 和給客戶端用的 client bundle。

server bundle運(yùn)行在node,所以代碼里面若出現(xiàn)window,document等瀏覽器對(duì)象則會(huì)報(bào)錯(cuò),可以引入jsdom解決,但是比較麻煩,還是建議用angular 官方推薦的方法

import { PLATFORM_ID } from '@angular/core';import { isPlatformBrowser, isPlatformServer } from '@angular/common'; constructor(@Inject(PLATFORM_ID) private platformId: Object) { ... } ngOnInit() { if (isPlatformBrowser(this.platformId)) {   // Client only code.   ... } if (isPlatformServer(this.platformId)) {  // Server only code.  ... }} 

通過(guò)PLATFORM_ID令牌注入的對(duì)象來(lái)檢查當(dāng)前平臺(tái)是瀏覽器還是服務(wù)器,從而解決該問(wèn)題。

client bundle運(yùn)行在瀏覽器,所以在這用使用瀏覽器對(duì)象就完全沒(méi)有問(wèn)題,但若涉及到像fs等node里才有的對(duì)象也會(huì)報(bào)錯(cuò),解決方案同上。

所以說(shuō)白了,server bundle就像是一個(gè)HTML文件的字符串,通過(guò)node渲染好后發(fā)送到前端,這個(gè)HTML字符串是可以同時(shí)運(yùn)行在node和瀏覽器的。

而 client bundle就像是一個(gè)js文件,我們前端里的所有事件都被包含在里面,我們可以在這里盡情地使用window對(duì)象。另外,盡管可以使用document對(duì)象,但是基于前端性能的考慮,還是不建議使用的。

 在使用angular這個(gè)項(xiàng)目中,開(kāi)發(fā)環(huán)境用JIT,生產(chǎn)環(huán)境用AOT,這個(gè)應(yīng)該是沒(méi)有爭(zhēng)論的。(具體AOT和JIT的區(qū)別 可參考https://angular.cn/docs/ts/latest/cookbook/aot-compiler.html#!#aot-jit

所以你也猜到了,我這個(gè)項(xiàng)目開(kāi)發(fā)環(huán)境是瀏覽器渲染+JIT,生產(chǎn)環(huán)境是服務(wù)端渲染+AOT。

所以主要步驟歸納如下:

1.  清除編譯文件目錄下的所有文件。

2.  執(zhí)行ngc分別預(yù)編譯客戶端代碼和服務(wù)端代碼,然后用webpack打包,壓縮

3.  node執(zhí)行編譯后的server bundle代碼

具體代碼實(shí)現(xiàn):

1、建立nodejs服務(wù)器,采用express框架(koa也是可以的),監(jiān)聽(tīng)端口

const express = require('express');const desktop = express();const port = process.env.NODE_PORT ? process.env.NODE_PORT : 4200;desktop.listen(port + 1, () => { console.log(`Desktop Listening on: http://localhost:${port}`);});

2、渲染頁(yè)面,處理請(qǐng)求

import 'zone.js/dist/zone-node';import { ngExpressEngine } from '@nguniversal/express-engine';import { AppServerModuleNgFactory } from './app-server.module.ngfactory';const ROOT = path.join(path.resolve(__dirname),'..','build'); function response(req, res) {  res.render(`index.html`, {   req,   res  }); } app.engine('html', ngExpressEngine({  bootstrap: AppServerModuleNgFactory })); app.set('view engine', 'html'); app.set('views', ROOT); app.get('/', response); routes.forEach((r) => {   app.get(r, response);   app.get(`${r}/*`, response);  } );

angular服務(wù)端渲染能一定程度上優(yōu)化用戶體驗(yàn),但是還是有個(gè)小問(wèn)題,用戶首次加載時(shí),仍然需要加載完整個(gè)網(wǎng)站的內(nèi)容。

所以我目前考慮在angular4 ssr的基礎(chǔ)上加入lazy load(懶惰加載,也稱(chēng)按需加載),懶惰加載模塊可幫助我們減少啟動(dòng)時(shí)間。通過(guò)懶惰加載,我們的應(yīng)用程序不需要一次加載所有內(nèi)容,只需要加載用戶在首次加載應(yīng)用程序時(shí)看到的內(nèi)容。

只有當(dāng)用戶導(dǎo)航到他們的路由時(shí),才會(huì)加載懶惰加載的模塊。以進(jìn)一步優(yōu)化用戶體驗(yàn),待完成后再寫(xiě)一遍隨筆記錄心路歷程。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持武林網(wǎng)。

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 元江| 满洲里市| 溧水县| 石首市| 宁海县| 板桥市| 普格县| 霍林郭勒市| 柘城县| 克山县| 两当县| 尖扎县| 宁国市| 柯坪县| 玉山县| 彭山县| 祥云县| 裕民县| 罗城| 江川县| 乌兰县| 巴南区| 祁阳县| 建阳市| 开平市| 湘西| 康定县| 克什克腾旗| 新营市| 仁怀市| 靖州| 大港区| 永城市| 兴仁县| 昆明市| 贵定县| 永康市| 惠安县| 陇川县| 汪清县| 慈利县|