作者:counterxing
鏈接:https://segmentfault.com/a/1190000018577041
隨著各大前端框架的誕生和演變, SPA
開始流行,單頁面應用的優(yōu)勢在于可以不重新加載整個頁面的情況下,通過 ajax
和服務器通信,實現(xiàn)整個 Web
應用拒不更新,帶來了極致的用戶體驗。然而,對于需要 SEO
、追求極致的首屏性能的應用,前端渲染的 SPA
是糟糕的。好在 Vue 2.0
后是支持服務端渲染的,零零散散花費了兩三周事件,通過改造現(xiàn)有項目,基本完成了在現(xiàn)有項目中實踐了 Vue
服務端渲染。
關于Vue服務端渲染的原理、搭建,官方文檔已經(jīng)講的比較詳細了,因此,本文不是抄襲文檔,而是文檔的補充。特別是對于如何與現(xiàn)有項目進行很好的結合,還是需要費很大功夫的。本文主要對我所在的項目中進行 Vue
服務端渲染的改造過程進行闡述,加上一些個人的理解,作為分享與學習。
本文主要分以下幾個方面:
什么是服務端渲染?服務端渲染的原理是什么?
如何在基于 Koa
的 Web Server Frame
上配置服務端渲染?
如何對現(xiàn)有項目進行改造?
在服務端預拉取數(shù)據(jù);
客戶端托管全局狀態(tài);
常見問題的解決方案;
基本目錄改造;
在服務端用 vue-router
分割代碼;
Vue.js
是構建客戶端應用程序的框架。默認情況下,可以在瀏覽器中輸出Vue
組件,進行生成DOM
和操作DOM
。然而,也可以將同一個組件渲染為服務器端的HTML
字符串,將它們直接發(fā)送到瀏覽器,最后將這些靜態(tài)標記'激活'為客戶端上完全可交互的應用程序。
上面這段話是源自Vue服務端渲染文檔的解釋,用通俗的話來說,大概可以這么理解:
服務端渲染的目的是:性能優(yōu)勢。 在服務端生成對應的 HTML
字符串,客戶端接收到對應的 HTML
字符串,能立即渲染 DOM
,最高效的首屏耗時。此外,由于服務端直接生成了對應的 HTML
字符串,對 SEO
也非常友好;
服務端渲染的本質是:生成應用程序的“快照”。將 Vue
及對應庫運行在服務端,此時, Web Server Frame
實際上是作為代理服務器去訪問接口服務器來預拉取數(shù)據(jù),從而將拉取到的數(shù)據(jù)作為 Vue
組件的初始狀態(tài)。
服務端渲染的原理是:虛擬 DOM
。在 Web Server Frame
作為代理服務器去訪問接口服務器來預拉取數(shù)據(jù)后,這是服務端初始化組件需要用到的數(shù)據(jù),此后,組件的beforeCreate
和 created
生命周期會在服務端調用,初始化對應的組件后, Vue
啟用虛擬 DOM
形成初始化的 HTML
字符串。之后,交由客戶端托管。實現(xiàn)前后端同構應用。
需要用到 Vue
服務端渲染對應庫 vue-server-renderer
,通過 npm
安裝:
npm install vue vue-server-renderer --save
最簡單的,首先渲染一個 Vue
實例:
// 第 1 步:創(chuàng)建一個 Vue 實例
const Vue = require('vue');
const app = new Vue({
template: `<div>Hello World</div>`
});
// 第 2 步:創(chuàng)建一個 renderer
const renderer = require('vue-server-renderer').createRenderer();
// 第 3 步:將 Vue 實例渲染為 HTML
renderer.renderToString(app, (err, html) => {
if (err) {
throw err;
}
console.log(html);
// => <div data-server-rendered='true'>Hello World</div>
});
與服務器集成:
module.exports = async function(ctx) {
ctx.status = 200;
let html = '';
try {
// ...
html = await renderer.renderToString(app, ctx);
} catch (err) {
ctx.logger('Vue SSR Render error', JSON.stringify(err));
html = await ctx.getErrorPage(err); // 渲染出錯的頁面
}
ctx.body = html;
}
使用頁面模板:
當你在渲染 Vue
應用程序時, renderer
只從應用程序生成 HTML
標記。在這個示例中,我們必須用一個額外的 HTML
頁面包裹容器,來包裹生成的 HTML
標記。
為了簡化這些,你可以直接在創(chuàng)建 renderer
時提供一個頁面模板。多數(shù)時候,我們會將頁面模板放在特有的文件中:
<!DOCTYPE html>
<html lang='en'>
<head><title>Hello</title></head>
<body>
<!--vue-ssr-outlet-->
</body>
</html>
然后,我們可以讀取和傳輸文件到 Vue renderer
中:
const tpl = fs.readFileSync(path.resolve(__dirname, './index.html'), 'utf-8');
const renderer = vssr.createRenderer({
template: tpl,
});
然而在實際項目中,不止上述例子那么簡單,需要考慮很多方面:路由、數(shù)據(jù)預取、組件化、全局狀態(tài)等,所以服務端渲染不是只用一個簡單的模板,然后加上使用 vue-server-renderer
完成的,如下面的示意圖所示:
如示意圖所示,一般的 Vue
服務端渲染項目,有兩個項目入口文件,分別為 entry-client.js
和 entry-server.js
,一個僅運行在客戶端,一個僅運行在服務端,經(jīng)過 Webpack
打包后,會生成兩個 Bundle
,服務端的 Bundle
會用于在服務端使用虛擬 DOM
生成應用程序的“快照”,客戶端的 Bundle
會在瀏覽器執(zhí)行。
因此,我們需要兩個 Webpack
配置,分別命名為 webpack.client.config.js
和 webpack.server.config.js
,分別用于生成客戶端 Bundle
與服務端 Bundle
,分別命名為 vue-ssr-client-manifest.json
與 vue-ssr-server-bundle.json
,關于如何配置, Vue
官方有相關示例vue-hackernews-2.0
我所在的項目使用 Koa
作為 Web Server Frame
,項目使用koa-webpack進行開發(fā)環(huán)境的構建。如果是在產(chǎn)品環(huán)境下,會生成 vue-ssr-client-manifest.json
與 vue-ssr-server-bundle.json
,包含對應的 Bundle
,提供客戶端和服務端引用,而在開發(fā)環(huán)境下,一般情況下放在內(nèi)存中。使用 memory-fs
模塊進行讀取。
const fs = require('fs')
const path = require( 'path' );
const webpack = require( 'webpack' );
const koaWpDevMiddleware = require( 'koa-webpack' );
const MFS = require('memory-fs');
const appSSR = require('./../../app.ssr.js');
let wpConfig;
let clientConfig, serverConfig;
let wpCompiler;
let clientCompiler, serverCompiler;
let clientManifest;
let bundle;
// 生成服務端bundle的webpack配置
if ((fs.existsSync(path.resolve(cwd,'webpack.server.config.js')))) {
serverConfig = require(path.resolve(cwd, 'webpack.server.config.js'));
serverCompiler = webpack( serverConfig );
}
// 生成客戶端clientManifest的webpack配置
if ((fs.existsSync(path.resolve(cwd,'webpack.client.config.js')))) {
clientConfig = require(path.resolve(cwd, 'webpack.client.config.js'));
clientCompiler = webpack(clientConfig);
}
if (serverCompiler && clientCompiler) {
let publicPath = clientCompiler.output && clientCompiler.output.publicPath;
const koaDevMiddleware = await koaWpDevMiddleware({
compiler: clientCompiler,
devMiddleware: {
publicPath,
serverSideRender: true
},
});
app.use(koaDevMiddleware);
// 服務端渲染生成clientManifest
app.use(async (ctx, next) => {
const stats = ctx.state.webpackStats.toJson();
const assetsByChunkName = stats.assetsByChunkName;
stats.errors.forEach(err => console.error(err));
stats.warnings.forEach(err => console.warn(err));
if (stats.errors.length) {
console.error(stats.errors);
return;
}
// 生成的clientManifest放到appSSR模塊,應用程序可以直接讀取
let fileSystem = koaDevMiddleware.devMiddleware.fileSystem;
clientManifest = JSON.parse(fileSystem.readFileSync(path.resolve(cwd,'./dist/vue-ssr-client-manifest.json'), 'utf-8'));
appSSR.clientManifest = clientManifest;
await next();
});
// 服務端渲染的server bundle 存儲到內(nèi)存里
const mfs = new MFS();
serverCompiler.outputFileSystem = mfs;
serverCompiler.watch({}, (err, stats) => {
if (err) {
throw err;
}
stats = stats.toJson();
if (stats.errors.length) {
console.error(stats.errors);
return;
}
// 生成的bundle放到appSSR模塊,應用程序可以直接讀取
bundle = JSON.parse(mfs.readFileSync(path.resolve(cwd,'./dist/vue-ssr-server-bundle.json'), 'utf-8'));
appSSR.bundle = bundle;
});
}
產(chǎn)品環(huán)境下,打包后的客戶端和服務端的 Bundle
會存儲為 vue-ssr-client-manifest.json
與 vue-ssr-server-bundle.json
,通過文件流模塊 fs
讀取即可,但在開發(fā)環(huán)境下,我創(chuàng)建了一個 appSSR
模塊,在發(fā)生代碼更改時,會觸發(fā) Webpack
熱更新, appSSR
對應的 bundle
也會更新, appSSR
模塊代碼如下所示:
let clientManifest;
let bundle;
const appSSR = {
get bundle() {
return bundle;
},
set bundle(val) {
bundle = val;
},
get clientManifest() {
return clientManifest;
},
set clientManifest(val) {
clientManifest = val;
}
};
module.exports = appSSR;
通過引入 appSSR
模塊,在開發(fā)環(huán)境下,就可以拿到 clientManifest
和 ssrBundle
,項目的渲染中間件如下:
const fs = require('fs');
const path = require('path');
const ejs = require('ejs');
const vue = require('vue');
const vssr = require('vue-server-renderer');
const createBundleRenderer = vssr.createBundleRenderer;
const dirname = process.cwd();
const env = process.env.RUN_ENVIRONMENT;
let bundle;
let clientManifest;
if (env === 'development') {
// 開發(fā)環(huán)境下,通過appSSR模塊,拿到clientManifest和ssrBundle
let appSSR = require('./../../core/app.ssr.js');
bundle = appSSR.bundle;
clientManifest = appSSR.clientManifest;
} else {
bundle = JSON.parse(fs.readFileSync(path.resolve(__dirname, './dist/vue-ssr-server-bundle.json'), 'utf-8'));
clientManifest = JSON.parse(fs.readFileSync(path.resolve(__dirname, './dist/vue-ssr-client-manifest.json'), 'utf-8'));
}
module.exports = async function(ctx) {
ctx.status = 200;
let html;
let context = await ctx.getTplContext();
ctx.logger('進入SSR,context為: ', JSON.stringify(context));
const tpl = fs.readFileSync(path.resolve(__dirname, './newTemplate.html'), 'utf-8');
const renderer = createBundleRenderer(bundle, {
runInNewContext: false,
template: tpl, // (可選)頁面模板
clientManifest: clientManifest // (可選)客戶端構建 manifest
});
ctx.logger('createBundleRenderer renderer:', JSON.stringify(renderer));
try {
html = await renderer.renderToString({
...context,
url: context.CTX.url,
});
} catch(err) {
ctx.logger('SSR renderToString 失?。?', JSON.stringify(err));
console.error(err);
}
ctx.body = html;
};
使用 Webpack
來處理服務器和客戶端的應用程序,大部分源碼可以使用通用方式編寫,可以使用 Webpack
支持的所有功能。
一個基本項目可能像是這樣:
src
├── components
│ ├── Foo.vue
│ ├── Bar.vue
│ └── Baz.vue
├── frame
│ ├── app.js # 通用 entry(universal entry)
│ ├── entry-client.js # 僅運行于瀏覽器
│ ├── entry-server.js # 僅運行于服務器
│ └── index.vue # 項目入口組件
├── pages
├── routers
└── store
app.js
是我們應用程序的「通用 entry
」。在純客戶端應用程序中,我們將在此文件中創(chuàng)建根 Vue
實例,并直接掛載到 DOM
。但是,對于服務器端渲染( SSR
),責任轉移到純客戶端 entry
文件。 app.js
簡單地使用 export
導出一個 createApp
函數(shù):
import Router from '~ut/router';
import { sync } from 'vuex-router-sync';
import Vue from 'vue';
import { createStore } from './../store';
import Frame from './index.vue';
import myRouter from './../routers/myRouter';
function createVueInstance(routes, ctx) {
const router = Router({
base: '/base',
mode: 'history',
routes: [routes],
});
const store = createStore({ ctx });
// 把路由注入到vuex中
sync(store, router);
const app = new Vue({
router,
render: function(h) {
return h(Frame);
},
store,
});
return { app, router, store };
}
module.exports = function createApp(ctx) {
return createVueInstance(myRouter, ctx);
}
注:在我所在的項目中,需要動態(tài)判斷是否需要注冊
DicomView
,只有在客戶端才初始化DicomView
,由于Node.js
環(huán)境沒有window
對象,對于代碼運行環(huán)境的判斷,可以通過typeof window === 'undefined'
來進行判斷。
如 Vue SSR
文檔所述:
當編寫純客戶端 (client-only) 代碼時,我們習慣于每次在新的上下文中對代碼進行取值。但是,Node.js 服務器是一個長期運行的進程。當我們的代碼進入該進程時,它將進行一次取值并留存在內(nèi)存中。這意味著如果創(chuàng)建一個單例對象,它將在每個傳入的請求之間共享。如基本示例所示,我們?yōu)槊總€請求創(chuàng)建一個新的根 Vue 實例。這與每個用戶在自己的瀏覽器中使用新應用程序的實例類似。如果我們在多個請求之間使用一個共享的實例,很容易導致交叉請求狀態(tài)污染 (cross-request state pollution)。因此,我們不應該直接創(chuàng)建一個應用程序實例,而是應該暴露一個可以重復執(zhí)行的工廠函數(shù),為每個請求創(chuàng)建新的應用程序實例。同樣的規(guī)則也適用于 router、store 和 event bus 實例。你不應該直接從模塊導出并將其導入到應用程序中,而是需要在 createApp 中創(chuàng)建一個新的實例,并從根 Vue 實例注入。
如上代碼所述, createApp
方法通過返回一個返回值創(chuàng)建 Vue
實例的對象的函數(shù)調用,在函數(shù) createVueInstance
中,為每一個請求創(chuàng)建了 Vue
, VueRouter
, Vuex
實例。并暴露給 entry-client
和 entry-server
模塊。
在客戶端 entry-client.js
只需創(chuàng)建應用程序,并且將其掛載到 DOM
中:
import { createApp } from './app';
// 客戶端特定引導邏輯……
const { app } = createApp();
// 這里假定 App.vue 模板中根元素具有 `id='app'`
app.$mount('#app');
服務端 entry-server.js
使用 default export
導出函數(shù),并在每次渲染中重復調用此函數(shù)。此時,除了創(chuàng)建和返回應用程序實例之外,它不會做太多事情 - 但是稍后我們將在此執(zhí)行服務器端路由匹配和數(shù)據(jù)預取邏輯:
import { createApp } from './app';
export default context => {
const { app } = createApp();
return app;
}
vue-router
分割代碼與 Vue
實例一樣,也需要創(chuàng)建單例的 vueRouter
對象。對于每個請求,都需要創(chuàng)建一個新的 vueRouter
實例:
function createVueInstance(routes, ctx) {
const router = Router({
base: '/base',
mode: 'history',
routes: [routes],
});
const store = createStore({ ctx });
// 把路由注入到vuex中
sync(store, router);
const app = new Vue({
router,
render: function(h) {
return h(Frame);
},
store,
});
return { app, router, store };
}
同時,需要在 entry-server.js
中實現(xiàn)服務器端路由邏輯,使用 router.getMatchedComponents
方法獲取到當前路由匹配的組件,如果當前路由沒有匹配到相應的組件,則 reject
到 404
頁面,否則 resolve
整個 app
,用于 Vue
渲染虛擬 DOM
,并使用對應模板生成對應的 HTML
字符串。
const createApp = require('./app');
module.exports = context => {
return new Promise((resolve, reject) => {
// ...
// 設置服務器端 router 的位置
router.push(context.url);
// 等到 router 將可能的異步組件和鉤子函數(shù)解析完
router.onReady(() => {
const matchedComponents = router.getMatchedComponents();
// 匹配不到的路由,執(zhí)行 reject 函數(shù),并返回 404
if (!matchedComponents.length) {
return reject('匹配不到的路由,執(zhí)行 reject 函數(shù),并返回 404');
}
// Promise 應該 resolve 應用程序實例,以便它可以渲染
resolve(app);
}, reject);
});
}
在 Vue
服務端渲染,本質上是在渲染我們應用程序的'快照',所以如果應用程序依賴于一些異步數(shù)據(jù),那么在開始渲染過程之前,需要先預取和解析好這些數(shù)據(jù)。服務端 WebServer Frame
作為代理服務器,在服務端對接口服務發(fā)起請求,并將數(shù)據(jù)拼裝到全局 Vuex
狀態(tài)中。
另一個需要關注的問題是在客戶端,在掛載到客戶端應用程序之前,需要獲取到與服務器端應用程序完全相同的數(shù)據(jù) - 否則,客戶端應用程序會因為使用與服務器端應用程序不同的狀態(tài),然后導致混合失敗。
目前較好的解決方案是,給路由匹配的一級子組件一個 asyncData
,在 asyncData
方法中, dispatch
對應的 action
。 asyncData
是我們約定的函數(shù)名,表示渲染組件需要預先執(zhí)行它獲取初始數(shù)據(jù),它返回一個 Promise
,以便我們在后端渲染的時候可以知道什么時候該操作完成。注意,由于此函數(shù)會在組件實例化之前調用,所以它無法訪問 this
。需要將 store
和路由信息作為參數(shù)傳遞進去:
舉個例子:
<!-- Lung.vue -->
<template>
<div></div>
</template>
<script>
export default {
// ...
async asyncData({ store, route }) {
return Promise.all([
store.dispatch('getA'),
store.dispatch('myModule/getB', { root:true }),
store.dispatch('myModule/getC', { root:true }),
store.dispatch('myModule/getD', { root:true }),
]);
},
// ...
}
</script>
在 entry-server.js
中,我們可以通過路由獲得與 router.getMatchedComponents()
相匹配的組件,如果組件暴露出 asyncData
,我們就調用這個方法。然后我們需要將解析完成的狀態(tài),附加到渲染上下文中。
const createApp = require('./app');
module.exports = context => {
return new Promise((resolve, reject) => {
const { app, router, store } = createApp(context);
// 針對沒有Vue router 的Vue實例,在項目中為列表頁,直接resolve app
if (!router) {
resolve(app);
}
// 設置服務器端 router 的位置
router.push(context.url.replace('/base', ''));
// 等到 router 將可能的異步組件和鉤子函數(shù)解析完
router.onReady(() => {
const matchedComponents = router.getMatchedComponents();
// 匹配不到的路由,執(zhí)行 reject 函數(shù),并返回 404
if (!matchedComponents.length) {
return reject('匹配不到的路由,執(zhí)行 reject 函數(shù),并返回 404');
}
Promise.all(matchedComponents.map(Component => {
if (Component.asyncData) {
return Component.asyncData({
store,
route: router.currentRoute,
});
}
})).then(() => {
// 在所有預取鉤子(preFetch hook) resolve 后,
// 我們的 store 現(xiàn)在已經(jīng)填充入渲染應用程序所需的狀態(tài)。
// 當我們將狀態(tài)附加到上下文,并且 `template` 選項用于 renderer 時,
// 狀態(tài)將自動序列化為 `window.__INITIAL_STATE__`,并注入 HTML。
context.state = store.state;
resolve(app);
}).catch(reject);
}, reject);
});
}
當服務端使用模板進行渲染時, context.state
將作為 window.__INITIAL_STATE__
狀態(tài),自動嵌入到最終的 HTML
中。而在客戶端,在掛載到應用程序之前, store
就應該獲取到狀態(tài),最終我們的 entry-client.js
被改造為如下所示:
import createApp from './app';
const { app, router, store } = createApp();
// 客戶端把初始化的store替換為window.__INITIAL_STATE__
if (window.__INITIAL_STATE__) {
store.replaceState(window.__INITIAL_STATE__);
}
if (router) {
router.onReady(() => {
app.$mount('#app')
});
} else {
app.$mount('#app');
}
至此,基本的代碼改造也已經(jīng)完成了,下面說的是一些常見問題的解決方案:
對于舊項目遷移到 SSR
肯定會經(jīng)歷的問題,一般為在項目入口處或是 created
、 beforeCreate
生命周期使用了 DOM
操作,或是獲取了 location
對象,通用的解決方案一般為判斷執(zhí)行環(huán)境,通過 typeof window
是否為 'undefined'
,如果遇到必須使用 location
對象的地方用于獲取 url
中的相關參數(shù),在 ctx
對象中也可以找到對應參數(shù)。
vue-router
報錯 Uncaught TypeError: _Vue.extend is not _Vue function
,沒有找到_Vue
實例的問題:
通過查看 Vue-router
源碼發(fā)現(xiàn)沒有手動調用 Vue.use(Vue-Router);
。沒有調用 Vue.use(Vue-Router);
在瀏覽器端沒有出現(xiàn)問題,但在服務端就會出現(xiàn)問題。對應的 Vue-router
源碼所示:
VueRouter.prototype.init = function init (app /* Vue component instance */) {
var this$1 = this;
process.env.NODE_ENV !== 'production' && assert(
install.installed,
'not installed. Make sure to call `Vue.use(VueRouter)` ' +
'before creating root instance.'
);
// ...
}
由于 hash
路由的參數(shù),會導致 vue-router
不起效果,對于使用了 vue-router
的前后端同構應用,必須換為 history
路由。
由于客戶端每次請求都會對應地把 cookie
帶給接口側,而服務端 Web ServerFrame
作為代理服務器,并不會每次維持 cookie
,所以需要我們手動把cookie
透傳給接口側,常用的解決方案是,將 ctx
掛載到全局狀態(tài)中,當發(fā)起異步請求時,手動帶上 cookie
,如下代碼所示:
// createStore.js
// 在創(chuàng)建全局狀態(tài)的函數(shù)`createStore`時,將`ctx`掛載到全局狀態(tài)
export function createStore({ ctx }) {
return new Vuex.Store({
state: {
...state,
ctx,
},
getters,
actions,
mutations,
modules: {
// ...
},
plugins: debug ? [createLogger()] : [],
});
}
當發(fā)起異步請求時,手動帶上 cookie
,項目中使用的是 Axios
:
// actions.js
// ...
const actions = {
async getUserInfo({ commit, state }) {
let requestParams = {
params: {
random: tool.createRandomString(8, true),
},
headers: {
'X-Requested-With': 'XMLHttpRequest',
},
};
// 手動帶上cookie
if (state.ctx.request.headers.cookie) {
requestParams.headers.Cookie = state.ctx.request.headers.cookie;
}
// ...
let res = await Axios.get(`${requestUrlOrigin}${url.GET_A}`, requestParams);
commit(globalTypes.SET_A, {
res: res.data,
});
}
};
// ...
接口請求時報 connect ECONNREFUSED 127.0.0.1:80
的問題
原因是改造之前,使用客戶端渲染時,使用了 devServer.proxy
代理配置來解決跨域問題,而服務端作為代理服務器對接口發(fā)起異步請求時,不會讀取對應的 webpack
配置,對于服務端而言會對應請求當前域下的對應 path
下的接口。
解決方案為去除 webpack
的 devServer.proxy
配置,對于接口請求帶上對應的 origin
即可:
const requestUrlOrigin = requestUrlOrigin = state.ctx.URL.origin;
const res = await Axios.get(`${requestUrlOrigin}${url.GET_A}`, requestParams);
對于 vue-router
配置項有 base
參數(shù)時,初始化時匹配不到對應路由的問題
在官方示例中的 entry-server.js
:
// entry-server.js
import { createApp } from './app';
export default context => {
// 因為有可能會是異步路由鉤子函數(shù)或組件,所以我們將返回一個 Promise,
// 以便服務器能夠等待所有的內(nèi)容在渲染前,
// 就已經(jīng)準備就緒。
return new Promise((resolve, reject) => {
const { app, router } = createApp();
// 設置服務器端 router 的位置
router.push(context.url);
// ...
});
}
原因是設置服務器端 router
的位置時, context.url
為訪問頁面的 url
,并帶上了 base
,在 router.push
時應該去除 base
,如下所示:
router.push(context.url.replace('/base', ''));
本文為筆者通過對現(xiàn)有項目進行改造,給現(xiàn)有項目加上 Vue
服務端渲染的實踐過程的總結。
首先闡述了什么是 Vue
服務端渲染,其目的、本質及原理,通過在服務端使用 Vue
的虛擬 DOM
,形成初始化的 HTML
字符串,即應用程序的“快照”。帶來極大的性能優(yōu)勢,包括 SEO
優(yōu)勢和首屏渲染的極速體驗。之后闡述了 Vue
服務端渲染的基本用法,即兩個入口、兩個 webpack
配置,分別作用于客戶端和服務端,分別生成 vue-ssr-client-manifest.json
與 vue-ssr-server-bundle.json
作為打包結果。最后通過對現(xiàn)有項目的改造過程,包括對路由進行改造、數(shù)據(jù)預獲取和狀態(tài)初始化,并解釋了在 Vue
服務端渲染項目改造過程中的常見問題,幫助我們進行現(xiàn)有項目往 Vue
服務端渲染的遷移。
聯(lián)系客服