成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

從零開始最小實現(xiàn) react 服務(wù)器渲染

cheukyin / 665人閱讀

摘要:從零開始最小實現(xiàn)服務(wù)器渲染前言最近在寫的時候想到,如果我部分代碼提供,部分代碼支持,那我應(yīng)該如何寫呢不想拆成個服務(wù)的情況下而且最近寫的項目里面也用過一些服務(wù)端渲染,如,自己也搭過的項目,確實開發(fā)體驗都非常友好,但是友好歸友好,具體又是如何實

從零開始最小實現(xiàn) react 服務(wù)器渲染 前言
最近在寫 koa 的時候想到,如果我部分代碼提供api,部分代碼支持ssr,那我應(yīng)該如何寫呢?(不想拆成 2個服務(wù)的情況下)

而且最近寫的項目里面也用過一些服務(wù)端渲染,如nuxt,自己也搭過next的項目,確實開發(fā)體驗都非常友好,但是友好歸友好,具體又是如何實現(xiàn)的呢,諸位有沒有考慮過?

本著求真務(wù)實的折騰態(tài)度,選了react作為研究對象(主要是vue寫的有點(diǎn)多,惡心了),那下面就簡單就以最小成本寫一個react的服務(wù)端渲染 demo

用到的技術(shù)棧

react 16 + webpack3 + koa2

看看它是如何實現(xiàn)服務(wù)端渲染的,here we go!

為什么要用服務(wù)端渲染 優(yōu)點(diǎn)

無非就是兩點(diǎn)

SEO 友好

加快首屏渲染,減少白屏?xí)r間

那么問題來了什么是SEO

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

而我們對SEO優(yōu)化,也是下面內(nèi)容所描述的核心就是:

下面是重點(diǎn)!

讓服務(wù)器把有內(nèi)容的HTML返回給我們,事件的話瀏覽器再渲染一次來進(jìn)行掛載
搭建 koa 環(huán)境

新建一個 ssr 項目,并在項目中初始化 npm

mkdir ssr && cd ssr
npm init

下面的代碼我們用到了 import jsx 等語法,node環(huán)境是不支持的,所以需要配置babel

在當(dāng)前項目中新建文件 server.jsindex.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("系統(tǒng)啟動,端口:3000")

根目錄下新建一個.babelrc文件
內(nèi)容是:

{
  "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",
}

到這里你運(yùn)行 npm run dev 打開localhost:3000

你就會看到 hello Koa

是不是很簡單就起了一個服務(wù)

安裝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 
hello world
} }

修改之前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()

  ctx.body = str
})

app.listen(3000)

console.log("系統(tǒng)啟動,端口:8080")

這個時候再 npm run dev
你就會看到屏幕上出現(xiàn)hello world

再打開chrome 開發(fā)者工具查看我們的請求:

我們的最簡單的react組件變成str傳了進(jìn)來

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

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

修改main.js的代碼

import React from "react"

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

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

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

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

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

說干就干

配制webpack

在根目錄下面新建一個 webpack.config.js
下面是webpack.config.js的內(nèi)容:

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設(shè)置成了app/index.js文件

那我們就創(chuàng)建一個

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

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

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

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

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

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

新增一點(diǎn)依賴

cnpm i --save koa-static koa-views ejs

koa-static: 處理靜態(tài)文件的中間件

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文件夾設(shè)置為靜態(tài)路徑
app.use(require("koa-static")(__dirname + "/public"))
// 將ejs設(shè)置為我們的模板引擎
app.use(views(path.resolve(__dirname, "./views"), { map: { html: "ejs" } }))

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

app.listen(3000)

console.log("系統(tǒng)啟動,端口:8080")

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

里面新建一個index.html:




    
    
    
    Document
    


    
<%- root %>

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

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

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

2. 運(yùn)行 npm run build, 構(gòu)建出我們的react代碼

3. npm run dev

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

 撒花,恭喜你,一個最簡單服務(wù)器渲染就已經(jīng)完成

到這里核心的思想就都已經(jīng)講完了,總結(jié)來說就下面三點(diǎn):

起一個node服務(wù)

react 根組件 renderToString渲染成字符串一起返回前端

前端再重新render一次

原理就是這么簡單

但是具體開發(fā)的時候還會有各種各樣的需求,比如:

不可能我每次改完代碼都重新構(gòu)建看效果吧 => 需要 實時構(gòu)建

create-react-app 都是熱更新,你還要刷新是不是太蠢了 => 需要支持熱更新

其他一些配套的周邊,如: react-router, redux 或者mobx怎么支持呢 => 需要完善的生態(tài)

.etc

這些問題都是用完 官方腳手架之后就回不去了的,所以更多的配置可以參考下面的repo(是一個工具鏈完善的最小實現(xiàn)),歡迎提PR

GitHub - ws456999/koa-react-ssr-starter: to understand && to explain how react ssr works

目前你可以在里面找到 react + react-router + mobx + postcss + 熱更新的配置,除了react-router的配置有些差別,其他都跟client端差別不大

That’s all.

參考鏈接

3-17 通過 Node.js API 啟動 Webpack · 深入淺出 Webpack
Koa2 + React + Webpack熱加載部署方案 | Nekoの喵窩
webpack - koa服務(wù)端熱更新問題 - SegmentFault
轉(zhuǎn) nodemon 基本配置與使用 - {前端開發(fā)} - 博客園
React 服務(wù)端渲染緩慢原因淺析 - 某熊的全棧之路 - SegmentFault
ReactDOMServer - React

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/92793.html

相關(guān)文章

  • 從零開始實現(xiàn)一個React(二):實現(xiàn)組件功能

    摘要:在這篇文章中,我們就要實現(xiàn)的組件功能。這篇文章的代碼從零開始實現(xiàn)系列是前端最受歡迎的框架之一,解讀其源碼的文章非常多,但是我想從另一個角度去解讀從零開始實現(xiàn)一個,從層面實現(xiàn)的大部分功能,在這個過程中去探索為什么有虛擬為什么這樣設(shè)計等問題。 前言 在上一篇文章JSX和虛擬DOM中,我們實現(xiàn)了基礎(chǔ)的JSX渲染功能,但是React的意義在于組件化。在這篇文章中,我們就要實現(xiàn)React的組件功...

    vslam 評論0 收藏0
  • 從零開始實現(xiàn)一個React(四):異步的setState

    摘要:一個比較好的做法是利用的事件隊列機(jī)制。整個系列大概會有四篇左右,我每周會更新一到兩篇,我會第一時間在上更新,有問題需要探討也請在上回復(fù)我博客地址關(guān)注點(diǎn),訂閱點(diǎn)上一篇文章從零開始實現(xiàn)一個三算法 前言 在上一篇文章中,我們實現(xiàn)了diff算法,性能有非常大的改進(jìn)。但是文章末尾也指出了一個問題:按照目前的實現(xiàn),每次調(diào)用setState都會觸發(fā)更新,如果組件內(nèi)執(zhí)行這樣一段代碼: for ( le...

    RyanQ 評論0 收藏0
  • 從零開始實現(xiàn)一個React(一):JSX和虛擬DOM

    摘要:前言是前端最受歡迎的框架之一,解讀其源碼的文章非常多,但是我想從另一個角度去解讀從零開始實現(xiàn)一個,從層面實現(xiàn)的大部分功能,在這個過程中去探索為什么有虛擬為什么這樣設(shè)計等問題。 前言 React是前端最受歡迎的框架之一,解讀其源碼的文章非常多,但是我想從另一個角度去解讀React:從零開始實現(xiàn)一個React,從API層面實現(xiàn)React的大部分功能,在這個過程中去探索為什么有虛擬DOM、d...

    曹金海 評論0 收藏0
  • 從零開始實現(xiàn)一個React(三):diff算法

    摘要:而對比變化,找出需要更新部分的算法我們稱之為算法。整個系列大概會有四篇,我每周會更新一到兩篇,我會第一時間在上更新,有問題需要探討也請在上回復(fù)我博客地址關(guān)注點(diǎn),訂閱點(diǎn)上一篇文章從零開始實現(xiàn)一個二組件和生命周期 前言 在上一篇文章,我們已經(jīng)實現(xiàn)了React的組件功能,從功能的角度來說已經(jīng)實現(xiàn)了React的核心功能了。 但是我們的實現(xiàn)方式有很大的問題:每次更新都重新渲染整個應(yīng)用或者整個組件...

    The question 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<