本文基于上一篇 angular5 的文章继续进行开发,上文中讲了搭建 angular5 有道翻译的过程,以及遇到问题的解决方案。
随后改了 ui,从 bootstrap4 改到 angular material,这里不详细讲,服务端渲染也与修改 ui 无关。
看过之前文章的人会发现,文章内容都偏向于服务端渲染,vue 的 nuxt,react 的 next。
在本次改版前也尝试去找类似 nuxt.js 与 next.js 的顶级封装库,可以大大节省时间,但是未果。
最后决定使用从 angular2 开始就可用的前后端同构解决方案 angular universal (universal (isomorphic) javascript support for angular.)
在这里不详细介绍文档内容,本文也尽量使用通俗易懂的语言带入 angular 的 ssr
前提
前面写的 udao 这个项目是完全遵从于 angular-cli 的,从搭建到打包,这也使得本文通用于所有 angular-cli 搭建的 angular5 项目。
搭建过程
首先安装服务端的依赖
yarn add @angular/platform-server expressyarn add -d ts-loader webpack-node-externals npm-run-all
这里需要注意的是 @angular/platform-server 的版本号最好根据当前 angular 版本进行安装,如: @angular/platform-server@5.1.0 ,避免与其它依赖有版本冲突。
创建文件: src/app/app.server.module.ts
import { ngmodule } from '@angular/core'import { servermodule } from '@angular/platform-server'import { appmodule } from './app.module'import { appcomponent } from './app.component'@ngmodule({ imports: [ appmodule, servermodule ], bootstrap: [appcomponent],})export class appservermodule { }
更新文件: src/app/app.module.ts
import { browsermodule } from '@angular/platform-browser'import { ngmodule } from '@angular/core'// ...import { appcomponent } from './app.component'// ...@ngmodule({ declarations: [ appcomponent // ... ], imports: [ browsermodule.withservertransition({ appid: 'udao' }) // ... ], providers: [], bootstrap: [appcomponent]})export class appmodule { }
我们需要一个主文件来导出服务端模块
创建文件: src/main.server.ts
export { appservermodule } from './app/app.server.module'
现在来更新 @angular/cli 的配置文件 .angular-cli.json
{ "$schema": "./node_modules/@angular/cli/lib/config/schema.json", "project": { "name": "udao" }, "apps": [ { "root": "src", "outdir": "dist/browser", "assets": [ "assets", "favicon.ico" ] // ... }, { "platform": "server", "root": "src", "outdir": "dist/server", "assets": [], "index": "index.html", "main": "main.server.ts", "test": "test.ts", "tsconfig": "tsconfig.server.json", "testtsconfig": "tsconfig.spec.json", "prefix": "app", "scripts": [], "environmentsource": "environments/environment.ts", "environments": { "dev": "environments/environment.ts", "prod": "environments/environment.prod.ts" } } ] // ...}
上面的 // ... 代表省略掉,但是 json 没有注释一说,看着怪怪的....
当然 .angular-cli.json 的配置不是固定的,根据需求自行修改
我们需要为服务端创建 tsconfig 配置文件: src/tsconfig.server.json
{ "extends": "../tsconfig.json", "compileroptions": { "outdir": "../out-tsc/app", "baseurl": "./", "module": "commonjs", "types": [] }, "exclude": [ "test.ts", "**/*.spec.ts", "server.ts" ], "angularcompileroptions": { "entrymodule": "app/app.server.module#appservermodule" }}
然后更新: src/tsconfig.app.json
{ "extends": "../tsconfig.json", "compileroptions": { "outdir": "../out-tsc/app", "baseurl": "./", "module": "es2015", "types": [] }, "exclude": [ "test.ts", "**/*.spec.ts", "server.ts" ]}
现在可以执行以下命令,看配置是否有效
ng build -prod --build-optimizer --app 0ng build --aot --app 1
运行结果应该如下图所示
然后就是创建 express.js 服务, 创建文件: src/server.ts
import 'reflect-metadata'import 'zone.js/dist/zone-node'import { rendermodulefactory } from '@angular/platform-server'import { enableprodmode } from '@angular/core'import * as express from 'express'import { join } from 'path'import { readfilesync } from 'fs'enableprodmode();const port = process.env.port || 4200const dist_folder = join(process.cwd(), 'dist')const app = express()const template = readfilesync(join(dist_folder, 'browser', 'index.html')).tostring()const { appservermodulengfactory } = require('main.server')app.engine('html', (_, options, callback) => { const opts = { document: template, url: options.req.url } rendermodulefactory(appservermodulengfactory, opts) .then(html => callback(null, html))});app.set('view engine', 'html')app.set('views', 'src')app.get('*.*', express.static(join(dist_folder, 'browser')))app.get('*', (req, res) => { res.render('index', { req })})app.listen(port, () => { console.log(`listening on http://localhost:${port}!`)})
理所当然需要一个 webpack 配置文件来打包 server.ts 文件: webpack.config.js
const path = require('path');var nodeexternals = require('webpack-node-externals');module.exports = { entry: { server: './src/server.ts' }, resolve: { extensions: ['.ts', '.js'], alias: { 'main.server': path.join(__dirname, 'dist', 'server', 'main.bundle.js') } }, target: 'node', externals: [nodeexternals()], output: { path: path.join(__dirname, 'dist'), filename: '[name].js' }, module: { rules: [ { test: /\.ts$/, loader: 'ts-loader' } ] }}
为了打包方便最好在 package.json 里面加几行脚本,如下:
"scripts": { "ng": "ng", "start": "ng serve", "build": "run-s build:client build:aot build:server", "build:client": "ng build -prod --build-optimizer --app 0", "build:aot": "ng build --aot --app 1", "build:server": "webpack -p", "test": "ng test", "lint": "ng lint", "e2e": "ng e2e"}
现在尝试运行 npm run build ,将会看到如下输出:
node 运行刚刚打包好的 node dist/server.js 文件
打开 http://localhost:4200/ 会正常显示项目主页面
从上面的开发者工具可以看出 html 文档是服务端渲染直出的,接下来尝试请求数据试一下。
注意:本项目显式(菜单可点击)的几个路由初始化都没有请求数据,但是单词解释的详情页是会在 ngoninit() 方法里获取数据,例如: http://localhost:4200/detail/add 直接打开时会发生奇怪的现象,请求在服务端和客户端分别发送一次,正常的服务端渲染项目首屏初始化数据的请求在服务端执行,在客户端不会二次请求!
发现问题后,就来踩平这个坑
试想如果采用一个标记来区分服务端是否已经拿到了数据,如果没拿到数据就在客户端请求,如果已经拿到数据就不发请求
当然 angular 早有一手准备,那就是 angular modules for transfer state
那么如何真实运用呢?见下文
请求填坑
在服务端入口和客户端入口分别引入 transferstatemodule
import { servermodule, servertransferstatemodule } from '@angular/platform-server';// ...@ngmodule({ imports: [ // ... servermodule, servertransferstatemodule ] // ...})export class appservermodule { }import { browsermodule, browsertransferstatemodule } from '@angular/platform-browser';// ...@ngmodule({ declarations: [ appcomponent // ... ], imports: [ browsermodule.withservertransition({ appid: 'udao' }), browsertransferstatemodule // ... ] // ...})export class appmodule { }
以本项目为例在 detail.component.ts 里面,修改如下
import { component, oninit } from '@angular/core'import { httpclient } from '@angular/common/http'import { router, activatedroute, navigationend } from '@angular/router'import { transferstate, makestatekey } from '@angular/platform-browser'const detail_key = makestatekey('detail')// ...export class detailcomponent implements oninit { details: any // some variable constructor( private http: httpclient, private state: transferstate, private route: activatedroute, private router: router ) {} transdata (res) { // translate res data } ngoninit () { this.details = this.state.get(detail_key, null as any) if (!this.details) { this.route.params.subscribe((params) => { this.loading = true const apiurl = `https://dict.youdao.com/jsonapi?q=${params['word']}` this.http.get(`/?url=${encodeuricomponent(apiurl)}`) .subscribe(res => { this.transdata(res) this.state.set(detail_key, res as any) this.loading = false }) }) } else { this.transdata(this.details) } }}
代码够简单清晰,和上面描述的原理一致
现在我们只需要对 main.ts 文件进行小小的调整,以便在 domcontentloaded 时运行我们的代码,以使 transferstate 正常工作:
import { enableprodmode } from '@angular/core'import { platformbrowserdynamic } from '@angular/platform-browser-dynamic'import { appmodule } from './app/app.module'import { environment } from './environments/environment'if (environment.production) { enableprodmode()}document.addeventlistener('domcontentloaded', () => { platformbrowserdynamic().bootstrapmodule(appmodule) .catch(err => console.log(err))})
到这里运行 npm run build && node dist/server.js 然后刷新 http://localhost:4200/detail/add 到控制台查看 network 如下:
发现 xhr 分类里面没有发起任何请求,只有 service-worker 的 cache 命中。
到这里坑都踩完了,项目运行正常,没发现其它 bug。
总结
2018 第一篇,目的就是探索所有流行框架服务端渲染的实现,开辟了 angular 这个最后没尝试的框架。
当然 orange 还是前端小学生一枚,只知道实现,原理说的不是很清楚,源码看的不是很明白,如有纰漏还望指教。
最后 github 地址和之前文章一样:https://github.com/orangexc/udao
上面是我整理给大家的,希望今后会对大家有帮助。
相关文章:
详解如何实现vuex(详细教程)
通过vue.js实现微信支付
在vue2.0中实现用户权限控制
以上就是使用angular5实现服务端渲染实战的详细内容。
