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

目标javascript

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

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

影响vue

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

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

安装webpack

1.nodejs 建议6+web

2.angular建议4.1+express

理论实现bootstrap

尽管这是一张来自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代码

 

具体代码实现

一、创建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}`);
});

 

二、渲染页面,处理请求

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(懒惰加载,也称按需加载),懒惰加载模块可帮助咱们减小启动时间。经过懒惰加载,咱们的应用程序不须要一次加载全部内容,只须要加载用户在首次加载应用程序时看到的内容。

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

相关文章
相关标签/搜索