react ssr 初體驗

用到的技術棧
react 16 + webpack3 + koa2

看看它是如何實現服務端渲染的,here we go!

為什么要用服務端渲染

優點

無非就是兩點

  1. SEO 友好
  2. 加快首屏渲染,減少白屏時間

那么問題來了什么是SEO

直接放文章不做贅述,前端后端分離,怎么解決SEO優化的問題呢? - 知乎
一句話介紹就是,現在我們做的大多是SPA網站,所有頁面啊數據啊都是ajax來的,搜索引擎的spider來收錄網頁的時候,發現全空?那么你覺得你的網站收錄的權重跟效果是好還是不好?

而我們對SEO優化,也是下面內容所描述的核心就是:

下面是重點!

讓服務器把有內容的HTML返回給我們,事件的話瀏覽器再渲染一次來進行掛載

搭建 koa 環境
新建一個 ssr 項目,并在項目中初始化 npm

mkdir ssr && cd ssr
npm init
下面的代碼我們用到了 import jsx 等語法,node環境是不支持的,所以需要配置babel

在當前項目中新建文件 server.js跟index.js,然后

babel的入口, index.js代碼如下

require('babel-core/register')()

require('babel-polyfill')
require('./server')
我們項目的入口, server.js代碼如下

import Koa from 'koa'
const app = new Koa()

// response
app.use((ctx) => {
ctx.body = 'Hello Koa'
})

app.listen(3000)
console.log("系統啟動,端口:3000")
根目錄下新建一個.babelrc文件
內容是:

{
"presets": ["react", "env"]
}
安裝上面所需要的依賴

npm install babel-core babel-polyfill babel-preset-env babel-preset-react nodemon --save-dev

npm i koa --save
配置啟動腳本

package.json

"scripts": {
"dev": "nodemon index.js",
}
到這里你運行 npm run dev 打開localhost:3000

你就會看到 hello Koa了

是不是很簡單就起了一個服務

安裝React
cnpm install react react-dom --save
在根目錄下新建一個app文件夾,并在文件夾中個新建一個main.js

main.js代碼如下

import React from 'react'

export default class Home extends React.Component {
render () {
return <div>hello world</div>
}
}
修改之前server.js

import Koa from 'koa'
import React from 'react'
import { renderToString } from 'react-dom/server'
import App from './app/main'

const app = new Koa()

// response
app.use(ctx => {
let str = renderToString(<App />)

ctx.body = str
})

app.listen(3000)

console.log('系統啟動,端口:8080')
這個時候再 npm run dev
你就會看到屏幕上出現hello world

這里我們用到了一個方法:
renderToString – 其實就是將組件渲染成字符串

目前為止,我們都還沒有給組件加上事件等交互行為,下面那讓我們來試一下

修改main.js的代碼

import React from 'react'

export default class Home extends React.Component {
  render () {
    return <div onClick={() => window.alert(123)}>hello world</div>
  }
}

再刷新一下我們的頁面,,咦,是不是沒有什么卵用

那是因為后端只能講組件渲染成一串html的字符串,事件綁定等事情都是需要在瀏覽器端執行的

那事件我們改怎么綁定上去呢?

那你肯定就會猜到,既然服務器渲染出來的是一串html,掛載事件的方式是不是在瀏覽器重新渲染一次就好了呢

說干就干


配制webpack

在根目錄下面新建一個 webpack.config.js
下面是webpack.config.js的內容:

var path = require('path')
var webpack = require('webpack')

module.exports = {
  entry: {
    main: './app/index.js'
  },
  output: {
    filename: '[name].js',
    path: path.join(__dirname, 'public'),
    publicPath: '/'
  },
  resolve: {
    extensions: ['.js', '.jsx']
  },
  module: {
    loaders: [
      {test: /\.jsx?$/,
        loaders: ['babel-loader'],
      }
    ]
  }
}

上面的配置將entry設置成了app/index.js文件

那我們就創建一個

下面是app/index.js的代碼:

import Demo from './main'
import ReactDOM from 'react-dom'
import React from 'react'
ReactDOM.render(<Demo />, document.getElementById('root'))

因為瀏覽器渲染需要將根組件掛載到某個dom節點上,所以給我們的react代碼設置一個入口

這個時候就有一個問題,就是,document對象node環境下并不存在,那怎么解決的呢?

不存在?不存在那我就不用就好了,SSR核心就是讓請求的url里面返回具體HTML內容,事件什么的并不care,那么我就把根組件直接renderToString
返回出來就好了唄

下面修改我們的服務代碼,讓代碼支持服務器渲染

新增一點依賴

cnpm i --save koa-static koa-views ejs
  • koa-static: 處理靜態文件的中間件
  • koa-views: 配置模板的中間件
  • ejs:一個模板引擎

修改server.js的代碼

import Koa from 'koa'
import React from 'react'
import { renderToString } from 'react-dom/server'
import views from 'koa-views'
import path from 'path'

import Demo from './app/main'
const app = new Koa()
// 將/public文件夾設置為靜態路徑
app.use(require('koa-static')(__dirname + '/public'))
// 將ejs設置為我們的模板引擎
app.use(views(path.resolve(__dirname, './views'), { map: { html: 'ejs' } }))

// response
app.use(async ctx => {
  let str = renderToString(<Demo />)
  await ctx.render('index', {
    root: str
  })
})

app.listen(3000)

console.log('系統啟動,端口:8080')

下面新建我們的渲染模板
新建一個views文件夾

里面新建一個index.html:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
    <base href="/client">
</head>
<body>
    <div id="root"><%- root %></div>
    <script src="/main.js"></script>
</body>
</html>

這個 html 里面可以放一些變量,比如這個<%- root %>,就是等下要放renderToString結果的地方
/main.js則是react構建出來的代碼

下面直接來測試一下我們的代碼

1. 在 package.json里面
新增:
  "scripts": {
    "dev": "nodemon index.js",
    "build": "webpack"
  },

2. 運行 npm run build, 構建出我們的react代碼

3. npm run dev

點擊一下代碼,是不是會 alert(123)

到這里核心的思想就都已經講完了,總結來說就下面三點:

  1. 起一個node服務
  2. react 根組件 renderToString渲染成字符串一起返回前端
  3. 前端再重新render一次
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,431評論 6 544
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,637評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,555評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,900評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,629評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,976評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,976評論 3 448
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,139評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,686評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,411評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,641評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,129評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,820評論 3 350
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,233評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,567評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,362評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,604評論 2 380

推薦閱讀更多精彩內容