详解基于Angular4+ server render(服务端渲染)开发教程


Posted in Javascript onAugust 28, 2017

目标:

1.更好的 SEO,方便搜索爬虫抓取页面内容

2.更快的内容到达时间(time-to-content)

影响:

1.用户:比原来更快的看到渲染的页面,提升用户体验

2.开发人员:某些代码可能需要特殊处理,才能在服务器渲染应用程序中运行(window,document, navigator等)

安装:

1.nodejs 建议6+

2.angular建议4.1+

理论实现:

详解基于Angular4+ server render(服务端渲染)开发教程

尽管这是一张来自vue官网服务器渲染的一张示意图,但是原理上和angular都是一样的,只是实现的代码不一致。

SSR 有两个入口文件,app-client.js 和 app-server.js, 都包含了应用代码(appmodule),webpack 通过两个入口文件分别打包成给服务端用的 server bundle 和给客户端用的 client bundle。

server bundle运行在node,所以代码里面若出现window,document等浏览器对象则会报错,可以引入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.

  ...

 }

}

通过PLATFORM_ID令牌注入的对象来检查当前平台是浏览器还是服务器,从而解决该问题。

client bundle运行在浏览器,所以在这用使用浏览器对象就完全没有问题,但若涉及到像fs等node里才有的对象也会报错,解决方案同上。

所以说白了,server bundle就像是一个HTML文件的字符串,通过node渲染好后发送到前端,这个HTML字符串是可以同时运行在node和浏览器的。

而 client bundle就像是一个js文件,我们前端里的所有事件都被包含在里面,我们可以在这里尽情地使用window对象。另外,尽管可以使用document对象,但是基于前端性能的考虑,还是不建议使用的。

 在使用angular这个项目中,开发环境用JIT,生产环境用AOT,这个应该是没有争论的。(具体AOT和JIT的区别 可参考https://angular.cn/docs/ts/latest/cookbook/aot-compiler.html#!#aot-jit)

所以你也猜到了,我这个项目开发环境是浏览器渲染+JIT,生产环境是服务端渲染+AOT。

所以主要步骤归纳如下:

1.  清除编译文件目录下的所有文件。

2.  执行ngc分别预编译客户端代码和服务端代码,然后用webpack打包,压缩

3.  node执行编译后的server bundle代码

具体代码实现:

1、建立nodejs服务器,采用express框架(koa也是可以的),监听端口

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、渲染页面,处理请求

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服务端渲染能一定程度上优化用户体验,但是还是有个小问题,用户首次加载时,仍然需要加载完整个网站的内容。

所以我目前考虑在angular4 ssr的基础上加入lazy load(懒惰加载,也称按需加载),懒惰加载模块可帮助我们减少启动时间。通过懒惰加载,我们的应用程序不需要一次加载所有内容,只需要加载用户在首次加载应用程序时看到的内容。

只有当用户导航到他们的路由时,才会加载懒惰加载的模块。以进一步优化用户体验,待完成后再写一遍随笔记录心路历程。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
jquery做的一个简单的屏幕锁定提示框
Mar 26 Javascript
jquery显示loading图片直到网页加载完成的方法
Jun 25 Javascript
bootstrap table 表格中增加下拉菜单末行出现滚动条的快速解决方法
Jan 05 Javascript
浅谈js for循环输出i为同一值的问题
Mar 01 Javascript
VUE实现表单元素双向绑定(总结)
Aug 08 Javascript
浅谈angularJS的$watch失效问题的解决方案
Aug 11 Javascript
vue实现页面加载动画效果
Sep 19 Javascript
vue组件发布到npm简单步骤
Nov 30 Javascript
vue-router 实现导航守卫(路由卫士)的实例代码
Sep 02 Javascript
js数组去重的方法总结
Jan 18 Javascript
稍微学一下Vue的数据响应式(Vue2及Vue3区别)
Nov 21 Javascript
基于jquery实现彩色投票进度条代码解析
Aug 26 jQuery
JS实现图片手风琴效果
Apr 17 #Javascript
vue服务端渲染的实例代码
Aug 28 #Javascript
jQuery 1.9版本以上的浏览器判断方法代码分享
Aug 28 #jQuery
使用react-router4.0实现重定向和404功能的方法
Aug 28 #Javascript
vue.js路由跳转详解
Aug 28 #Javascript
jQuery Collapse1.1.0折叠插件简单使用
Aug 28 #jQuery
解决IE7中使用jQuery动态操作name问题
Aug 28 #jQuery
You might like
默默简单的写了一个模板引擎
2007/01/02 PHP
如何设置mysql允许外网访问
2013/06/04 PHP
Laravel实现定时任务的示例代码
2017/08/10 PHP
JS 动态加载脚本的4种方法
2009/05/05 Javascript
Jquery为a标签的href赋值实现代码
2013/05/03 Javascript
jquery中插件实现自动添加用户的具体代码
2013/11/15 Javascript
js取模(求余数)隔行变色
2014/05/15 Javascript
javascript中Function类型详解
2015/04/28 Javascript
JS+DIV+CSS实现的经典标签切换效果代码
2015/09/14 Javascript
JavaScript函数学习总结以及相关的编程习惯指南
2015/11/16 Javascript
Bootstrap每天必学之滚动监听
2016/03/16 Javascript
Bootstrap 最常用的JS插件系列总结(图片轮播、标签切换等)
2016/07/14 Javascript
基于javascript实现按圆形排列DIV元素(一)
2016/12/02 Javascript
jQuery实现简易的输入框字数计数功能示例
2017/01/16 Javascript
JS设计模式之数据访问对象模式的实例讲解
2017/09/30 Javascript
使用nvm管理不同版本的node与npm的方法
2017/10/31 Javascript
详解Vue单元测试case写法
2018/05/24 Javascript
webpack4与babel配合使es6代码可运行于低版本浏览器的方法
2018/10/12 Javascript
layui多图上传实现删除功能的例子
2019/09/23 Javascript
vue axios请求成功却进入catch的原因分析
2020/09/08 Javascript
Javascript 模拟mvc实现点餐程序案例详解
2020/12/24 Javascript
[01:29:31]VP VS VG Supermajor小组赛胜者组第二轮 BO3第一场 6.2
2018/06/03 DOTA
Python爬取网页中的图片(搜狗图片)详解
2017/03/23 Python
python字符串的方法与操作大全
2018/01/30 Python
python实现简单图片物体标注工具
2019/03/18 Python
python如何获取列表中每个元素的下标位置
2019/07/01 Python
解决Django 在ForeignKey中出现 non-nullable field错误的问题
2019/08/06 Python
pytorch 状态字典:state_dict使用详解
2020/01/17 Python
jupyter notebook实现显示行号
2020/04/13 Python
美国美发品牌:Bumble and Bumble
2016/10/08 全球购物
Linux中如何用命令创建目录
2015/01/12 面试题
《陋室铭》教学反思
2014/02/26 职场文书
给学校的建议书
2014/03/12 职场文书
初中班主任评语
2014/04/24 职场文书
李敖北大演讲稿
2014/05/24 职场文书
六年级上册《闻官军收河南河北》的教学设计
2019/11/15 职场文书