欧美亚洲精品一区,日韩电影中文字幕一区,欧美日韩国产一二三,美女100%一区

當前位置: 首頁 / 新聞 / 網(wǎng)站知識 / 正文

恩尼斯網(wǎng)站開發(fā)定制網(wǎng)站小教程(一):Vue 服務端是如何渲染(SSR)

來源:qiuzhx時間:2019-09-18

服務端渲染的模式就是:就是將本來要放在瀏覽器執(zhí)行創(chuàng)建的組件,放到服務端先創(chuàng)建好,然后生成對應的html,在請求一個網(wǎng)址的時候,服務端收到請求之后把html的內(nèi)容先生成好然后再返回給瀏覽器。這樣子搜索引擎就可以通過你返回的a標簽抓取到網(wǎng)站的其他頁面了,最后將這些靜態(tài)標記"激活"為客戶端上完全可交互的應用程序


Vue SSR 相比 SPA(單頁應用)好處及壞處
好處

更好的 SEO,由于搜索引擎爬蟲抓取工具可以直接查看完全渲染的頁面。
更快的內(nèi)容到達時間(time-to-content),特別是對于緩慢的網(wǎng)絡情況或運行緩慢的設備。
壞處

開發(fā)條件所限。瀏覽器特定的代碼,只能在某些生命周期鉤子函數(shù)(lifecycle hook)中使用;一些外部擴展庫(external library)可能需要特殊處理,才能在服務器渲染應用程序中運行。
涉及構建設置和部署的更多要求。與可以部署在任何靜態(tài)文件服務器上的完全靜態(tài)單頁面應用程序(SPA)不同,服務器渲染應用程序,需要處于 Node.js server 運行環(huán)境。
更多的服務器端負載。在 Node.js 中渲染完整的應用程序,顯然會比僅僅提供靜態(tài)文件的 server 更加大量占用 CPU 資源(CPU-intensive - CPU 密集),因此如果你預料在高流量環(huán)境(high traffic)下使用,請準備相應的服務器負載,并明智地采用緩存策略。
 

 

SSR原理
這是vue.js官方的SSR原理介紹圖,從這張圖片,我們可以知道:我們需要通過Webpack打包生成兩份bundle文件:

Client Bundle,給瀏覽器用。和純Vue前端項目Bundle類似
Server Bundle,供服務端SSR使用,一個json文件
不管你項目先前是什么樣子,是否是使用vue-cli生成的。都會有這個構建改造過程。在構建改造這里會用到 vue-server-renderer 庫,這里要注意的是 vue-server-renderer 版本要與Vue版本一樣。

Vue SSR實例(構建基于vue-cli3 的SSR應用程序)
創(chuàng)建一個vue項目

vue create ssr-example 
............這里就i不再過多的說明了,就正常的創(chuàng)建一個vue項目的流程

進行SSR改造—安裝需要的包

安裝 vue-server-renderer
安裝 lodash.merge
安裝 webpack-node-externals
安裝 cross-env
npm install vue-server-renderer lodash.merge webpack-node-externals cross-env --save-dev
進行SSR改造—在服務器中集成

安裝koa2,然后在項目根目錄下新建一個server.js

npm install koa koa-static --save
進行SSR改造—在koa2集成ssr

源碼:

----------------------------------------------------------------------------------

// 第 1 步:創(chuàng)建一個 Vue 實例
const Vue = require('vue')
 
const Koa = require('koa')
const app = new Koa()
// 第 2 步:創(chuàng)建一個 renderer
const renderer = require('vue-server-renderer').createRenderer()
 
 
// 第 3 步:添加一個中間件來處理所有請求
app.use(async (ctx, next) => {
  const vm = new Vue({
    data: {
      title: "ssr example",
      url: ctx.url
    },
    template: `<div>訪問的 URL 是: {{ url }}</div>`
  })
  // 將 Vue 實例渲染為 HTML
  renderer.renderToString(vm, (err, html) => {
    if(err){
      res.status(500).end('Internal Server Error')
      return
    }
    ctx.body = html
  })
})
 
const port = 3000
app.listen(port, function() {
  console.log(`server started at localhost:${port}`)
})
--------------------------------------------------------------------------------------

看到這個說明一個簡單的ssr構建成功了。

不過到目前為止,我們并沒有將客戶端的.vue文件通過服務端進行渲染,那么如何將前端的.vue文件與后端node進行結合呢?

進行SSR改造—改造前端配置,以支持SSR

修改源碼結構

在src目錄下新建兩個文件,一個entry-client.js 僅運行于瀏覽器 一個entry-server.js 僅運行于服務器
修改main.js
main.js 是我們應用程序的「通用 entry」。在純客戶端應用程序中,我們將在此文件中創(chuàng)建根 Vue 實例,并直接掛載到 DOM。但是,對于服務器端渲染(SSR),責任轉移到純客戶端 entry 文件。app.js 簡單地使用 export 導出一個 createApp 函數(shù):

import Vue from 'vue'
import App from './App.vue'
import { createRouter } from "./router"
import store from './store'
 
Vue.config.productionTip = false
 
// 導出一個工廠函數(shù),用于創(chuàng)建新的
// 應用程序、router 和 store 實例
export function createApp () {
  const router = createRouter()
  const app = new Vue({
    router,
    store,
    // 根實例簡單的渲染應用程序組件。
    render: h => h(App)
  })
  return { app }
}
修改entry-client.js

客戶端 entry 只需創(chuàng)建應用程序,并且將其掛載到 DOM 中

import { createApp } from './main'
 
// 客戶端特定引導邏輯……
const { app } = createApp()
 
// 這里假定 App.vue 模板中根元素具有 `id="app"`
app.$mount('#app')
修改entry-server.js

服務器 entry 使用 default export 導出函數(shù),并在每次渲染中重復調用此函數(shù)。

import { createApp } from './main'
 
export default context => {
  // 因為有可能會是異步路由鉤子函數(shù)或組件,所以我們將返回一個 Promise,
  // 以便服務器能夠等待所有的內(nèi)容在渲染前,
  // 就已經(jīng)準備就緒。
  return new Promise((resolve, reject) => {
    const { app, router, store } = createApp()
 
    // 設置服務器端 router 的位置
    router.push(context.url)
 
    // 等到 router 將可能的異步組件和鉤子函數(shù)解析完
    router.onReady(() => {
      const matchedComponents = router.getMatchedComponents()
      // 匹配不到的路由,執(zhí)行 reject 函數(shù),并返回 404
      if (!matchedComponents.length) {
        return reject({
          code: 404
        })
      }
      // Promise 應該 resolve 應用程序實例,以便它可以渲染
      resolve(app)
    }, reject)
  })
}
修改router.js

import Vue from 'vue'
import Router from 'vue-router'
import Home from './views/Home.vue'
 
Vue.use(Router)
 
export default new Router({
  mode: 'history', // 一定要是history模式
  base: process.env.BASE_URL,
  routes: [
    {
      path: '/',
      name: 'home',
      component: Home
    },
    {
      path: '/about',
      name: 'about',
      // route level code-splitting
      // this generates a separate chunk (about.[hash].js) for this route
      // which is lazy-loaded when the route is visited.
      component: () => import(/* webpackChunkName: "about" */ './views/About.vue')
    }
  ]
})

修改webpack配置

在vue-cli3創(chuàng)建的vue項目,已經(jīng)沒有了之前的webpack.base.conf.js、webpack.dev.conf.js、webpack.prod.conf.js。那么如何進行webpack的配置呢?

在vue-cli官網(wǎng)上也說明了如何使用。 調整 webpack 配置最簡單的方式就是在 vue.config.js 中的 configureWebpack 選項提供一個對象,該對象將會被 webpack-merge 合并入最終的 webpack 配置。

在項目根目錄下,新建一個vue.config.js

const VueSSRServerPlugin = require("vue-server-renderer/server-plugin")
const VueSSRClientPlugin = require("vue-server-renderer/client-plugin")
const nodeExternals = require("webpack-node-externals")
const merge = require("lodash.merge")
const TARGET_NODE = process.env.WEBPACK_TARGET === "node"
const target = TARGET_NODE ? "server" : "client"
 
 
module.exports = {
  configureWebpack: () => ({
    // 將 entry 指向應用程序的 server / client 文件
    entry: `./src/entry-${target}.js`,
    // 對 bundle renderer 提供 source map 支持
    devtool: 'source-map',
    target: TARGET_NODE ? "node" : "web",
    node: TARGET_NODE ? undefined : false,
    output: {
      libraryTarget: TARGET_NODE ? "commonjs2" : undefined
    },
    // https://webpack.js.org/configuration/externals/#function
    // https://github.com/liady/webpack-node-externals
    // 外置化應用程序依賴模塊。可以使服務器構建速度更快,
    // 并生成較小的 bundle 文件。
    externals: nodeExternals({
      // 不要外置化 webpack 需要處理的依賴模塊。
      // 你可以在這里添加更多的文件類型。例如,未處理 *.vue 原始文件,
      // 你還應該將修改 `global`(例如 polyfill)的依賴模塊列入白名單
      whitelist: [/.css$/]
    }),
    optimization: {
      splitChunks: {
        chunks: "async",
        minSize: 30000,
        minChunks: 2,
        maxAsyncRequests: 5,
        maxInitialRequests: 3
      }
    },
    plugins: [TARGET_NODE ? new VueSSRServerPlugin() : new VueSSRClientPlugin()]
  }),
  chainWebpack: config => {
    config.module
      .rule("vue")
      .use("vue-loader")
      .tap(options => {
        merge(options, {
          optimizeSSR: false
        })
      })
  }
}
修改package,在scripts下面新增三個腳本來生成bundle.json

"build:client": "vue-cli-service build",
"build:server": "cross-env WEBPACK_TARGET=node vue-cli-service build",
"build:win": "npm run build:server && move dist\\vue-ssr-server-bundle.json bundle && npm run build:client && move bundle dist\\vue-ssr-server-bundle.json",
執(zhí)行命令

npm run build:win
在dist目錄下會生成兩個json文件



改造server.js 代碼

const fs = require('fs')
const Koa = require('koa')
const path = require('path')
const koaStatic = require('koa-static')
const app = new Koa()
 
const resolve = file => path.resolve(__dirname, file)
// 開放dist目錄
app.use(koaStatic(resolve('./dist')))
 
// 第 2 步:獲得一個createBundleRenderer
const { createBundleRenderer } = require('vue-server-renderer')
const bundle = require('./dist/vue-ssr-server-bundle.json')
const clientManifest = require('./dist/vue-ssr-client-manifest.json')
 
const renderer = createBundleRenderer(bundle, {
  runInNewContext: false,
  template: fs.readFileSync(resolve('./src/index.temp.html'), 'utf-8'),
  clientManifest: clientManifest
})
 
function renderToString(context) {
  return new Promise((resolve, reject) => {
    renderer.renderToString(context, (err, html) => {
      err ? reject(err) : resolve(html)
    })
  })
}
// 第 3 步:添加一個中間件來處理所有請求
app.use(async (ctx, next) => {
  const context = {
    title: 'ssr test',
    url: ctx.url
  }
  // 將 context 數(shù)據(jù)渲染為 HTML
  const html = await renderToString(context)
  ctx.body = html
})
 
const port = 3000
app.listen(port, function() {
  console.log(`server started at localhost:${port}`)
})
運行server.js

node server.js
訪問 localhost:3000,可以看到頁面的數(shù)據(jù)都是又服務端渲染完成后返回的。到這一步已經(jīng)基本算完成了SSR的構建了。

如果有問題的話,可以把dist目錄下的index.html文件刪了。避免直接訪問到了該html文件

恩尼斯網(wǎng)站開發(fā)定制網(wǎng)站轉載

欧美亚洲精品一区,日韩电影中文字幕一区,欧美日韩国产一二三,美女100%一区
aaa亚洲精品| 久久综合九色综合97婷婷 | 一区二区三区在线免费| 五月天丁香久久| 国产ts人妖一区二区| 欧美精品亚洲一区二区在线播放| 久久夜色精品一区| 亚洲成人精品一区二区| av一区二区久久| 91精品国产欧美一区二区| 亚洲三级在线免费| 国内偷窥港台综合视频在线播放| 在线国产亚洲欧美| 欧美国产精品一区二区三区| 日韩av一级片| 欧美色图片你懂的| 最新久久zyz资源站| 国产激情视频一区二区三区欧美| 日韩一区二区三区视频| 一区二区三区四区不卡在线| 国产 日韩 欧美大片| 久久夜色精品一区| 麻豆精品在线视频| 欧美一区二区三区婷婷月色| 亚洲图片一区二区| 一本色道久久综合亚洲91| 亚洲国产高清在线| 风流少妇一区二区| 亚洲国产精品成人综合色在线婷婷| 免费成人深夜小野草| 88在线观看91蜜桃国自产| 香蕉久久夜色精品国产使用方法 | 国产精品三级av在线播放| 国产一区二区精品在线观看| 日韩欧美电影一区| 免费观看久久久4p| 欧美一区中文字幕| 蜜臀久久99精品久久久久宅男| 91精品国产综合久久香蕉麻豆| 无吗不卡中文字幕| 91麻豆精品91久久久久同性| 日本不卡123| 日韩视频免费观看高清完整版 | 亚洲免费三区一区二区| 色94色欧美sute亚洲线路一ni| 日韩美女啊v在线免费观看| 色视频欧美一区二区三区| 亚洲综合无码一区二区| 欧美巨大另类极品videosbest | 日韩欧美成人一区| 国产一区二区三区最好精华液| 久久精品日韩一区二区三区| 粉嫩高潮美女一区二区三区| 中文字幕一区三区| 欧美日韩国产成人在线免费| 日韩高清在线一区| 久久久五月婷婷| 成人av电影在线网| 亚洲电影第三页| 欧美mv日韩mv亚洲| 成人免费三级在线| 午夜免费久久看| 久久久久久久久久久久久夜| 91丨porny丨首页| 日韩黄色免费电影| 国产亚洲午夜高清国产拍精品| 91麻豆免费在线观看| 秋霞午夜av一区二区三区 | 欧美岛国在线观看| 丁香婷婷综合五月| 亚洲一区二区黄色| 久久久激情视频| 欧美乱妇15p| 成人精品高清在线| 日韩中文字幕区一区有砖一区| 国产欧美日韩综合| 在线观看91精品国产麻豆| 国产毛片精品国产一区二区三区| 亚洲色欲色欲www在线观看| 制服丝袜亚洲色图| 99精品视频在线观看| 免费在线观看一区| 亚洲婷婷综合久久一本伊一区| 日韩欧美一区中文| 色综合久久九月婷婷色综合| 国产精品综合av一区二区国产馆| 亚洲永久精品大片| 中文字幕av在线一区二区三区| 在线不卡的av| 欧美天天综合网| av一区二区不卡| 国产电影精品久久禁18| 日韩成人午夜精品| 一二三四社区欧美黄| 国产精品美女久久久久aⅴ| 日韩网站在线看片你懂的| 色爱区综合激月婷婷| 不卡av电影在线播放| 国产精品 欧美精品| 另类小说图片综合网| 婷婷成人综合网| 亚洲v中文字幕| 亚洲永久免费av| 亚洲欧美激情在线| 国产精品福利一区二区| 久久久久久9999| 久久品道一品道久久精品| 欧美一卡在线观看| 日韩小视频在线观看专区| 欧美福利视频导航| 欧美一区二区三区婷婷月色| 欧美一区二区三区男人的天堂| 欧美日韩精品一区二区三区四区| 91色|porny| 色噜噜狠狠成人中文综合| 成人av网站免费| 成人听书哪个软件好| 高清不卡一二三区| 国产a精品视频| caoporn国产精品| 99久久99久久久精品齐齐| 一本到高清视频免费精品| 91年精品国产| 欧美无人高清视频在线观看| 欧美日韩三级一区二区| 制服丝袜中文字幕亚洲| 日韩久久久精品| 国产午夜精品福利| 国产精品久久久久影院亚瑟 | 在线观看一区二区视频| 在线国产电影不卡| 欧美日韩精品一区二区| 日韩欧美国产综合一区 | 国产乱码精品一区二区三区五月婷 | 亚洲欧美视频在线观看视频| 亚洲精品综合在线| 婷婷综合在线观看| 国产剧情一区二区三区| 成人一区二区三区视频| 精品视频资源站| wwwwww.欧美系列| 国产精品免费视频网站| 亚洲综合一区在线| 国产主播一区二区三区| 91蜜桃在线免费视频| 欧美日韩国产精品自在自线| 2014亚洲片线观看视频免费| 亚洲色图.com| 国内一区二区在线| www.性欧美| 在线不卡免费欧美| 中文字幕乱码亚洲精品一区| 午夜精品一区在线观看| 国产一区二区三区免费观看| 在线国产亚洲欧美| 久久精品夜色噜噜亚洲a∨| 亚洲午夜久久久久| 成人动漫在线一区| 欧美一区欧美二区| 亚洲视频中文字幕| 黄网站免费久久| 欧美日韩精品二区第二页| 中文一区在线播放| 蜜臂av日日欢夜夜爽一区| 国产乱码精品一区二区三区忘忧草| 欧美无砖砖区免费| 国产精品高清亚洲| 国产精品主播直播| 日韩一级高清毛片| 一区二区三区电影在线播| 高清在线观看日韩| 久久久精品中文字幕麻豆发布| 午夜久久电影网| 日本乱人伦一区| 国产欧美综合色| 精品一区免费av| 7777精品伊人久久久大香线蕉| 亚洲欧美日韩小说| 成人国产精品免费观看动漫| 2020国产精品自拍| 麻豆国产一区二区| 日韩一区二区三区在线观看 | 亚洲自拍另类综合| 菠萝蜜视频在线观看一区| 国产视频一区二区三区在线观看| 六月丁香综合在线视频| 91精品免费在线| 日韩高清一区在线| 在线电影欧美成精品| 亚洲电影一级黄| 欧美日韩在线观看一区二区 | 亚洲欧美在线视频| 成人黄色777网| 国产精品免费av| 成人免费av资源| 国产精品久久久久久亚洲伦| 成人黄色小视频| 亚洲日本在线观看| 色婷婷香蕉在线一区二区| 亚洲男女一区二区三区|