因為對網(wǎng)頁SEO的需要,要把之前的React項目改造為服務(wù)端渲染,經(jīng)過一番調(diào)查和研究,查閱了大量互聯(lián)網(wǎng)資料。成功踩坑。
創(chuàng)新互聯(lián)專業(yè)為企業(yè)提供張店網(wǎng)站建設(shè)、張店做網(wǎng)站、張店網(wǎng)站設(shè)計、張店網(wǎng)站制作等企業(yè)網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計與制作、張店企業(yè)網(wǎng)站模板建站服務(wù),十年張店做網(wǎng)站經(jīng)驗,不只是建網(wǎng)站,更提供有價值的思路和整體網(wǎng)絡(luò)服務(wù)。
選型思路:實現(xiàn)服務(wù)端渲染,想用React最新的版本,并且不對現(xiàn)有的寫法做大的改動,如果一開始就打算服務(wù)端渲染,建議直接用NEXT框架來寫
項目地址:https://github.com/wlx200510/react_koa_ssr
腳手架選型:webpack3.11.0 + react Router4 + Redux + koa2 + React16 + Node8.x
主要心得:對React的相關(guān)知識更加熟悉,成功拓展自己的技術(shù)領(lǐng)域,對服務(wù)端技術(shù)在實際項目上有所積累
注意點:使用框架前一定確認當前webpack版本為3.x Node為8.x以上,讀者最好用React在3個月以上,并有實際React項目經(jīng)驗
項目目錄介紹
├── assets │ └── index.css //放置一些全局的資源文件 可以是js 圖片等 ├── config │ ├── webpack.config.dev.js 開發(fā)環(huán)境webpack打包設(shè)置 │ └── webpack.config.prod.js 生產(chǎn)環(huán)境webpack打包設(shè)置 ├── package.json ├── README.md ├── server server端渲染文件,如果對不是很了解,建議參考[koa教程](http://wlxadyl.cn/2018/02/11/koa-learn/) │ ├── app.js │ ├── clientRouter.js // 在此文件中包含了把服務(wù)端路由匹配到react路由的邏輯 │ ├── ignore.js │ └── index.js └── src ├── app 此文件夾下主要用于放置瀏覽器和服務(wù)端通用邏輯 │ ├── configureStore.js //redux-thunk設(shè)置 │ ├── createApp.js //根據(jù)渲染環(huán)境不同來設(shè)置不同的router模式 │ ├── index.js │ └── router │ ├── index.js │ └── routes.js //路由配置文件! 重要 ├── assets │ ├── css 放置一些公共的樣式文件 │ │ ├── _base.scss //很多項目都會用到的初始化css │ │ ├── index.scss │ │ └── my.scss │ └── img ├── components 放置一些公共的組件 │ ├── FloatDownloadBtn 公共組件樣例寫法 │ │ ├── FloatDownloadBtn.js │ │ ├── FloatDownloadBtn.scss │ │ └── index.js │ ├── Loading.js │ └── Model.js 函數(shù)式組件的寫法 │ ├── favicon.ico ├── index.ejs //渲染的模板 如果項目需要,可以放一些公共文件進去 ├── index.js //包括熱更新的邏輯 ├── pages 頁面組件文件夾 │ ├── home │ │ ├── components // 用于放置頁面組件,主要邏輯 │ │ │ └── homePage.js │ │ ├── containers // 使用connect來封裝出高階組件 注入全局state數(shù)據(jù) │ │ │ └── homeContainer.js │ │ ├── index.js // 頁面路由配置文件 注意thunk屬性 │ │ └── reducer │ │ └── index.js // 頁面的reducer 這里暴露出來給store統(tǒng)一處理 注意寫法 │ └── user │ ├── components │ │ └── userPage.js │ ├── containers │ │ └── userContainer.js │ └── index.js └── store ├── actions // 各action存放地 │ ├── home.js │ └── thunk.js ├── constants.js // 各action名稱匯集處 防止重名 └── reducers └── index.js // 引用各頁面的所有reducer 在此處統(tǒng)一combine處理
項目的構(gòu)建思路
本地開發(fā)介紹
查看本地開發(fā)主要涉及的文件是src目錄下的index.js文件,判斷當前的運行環(huán)境,只有在開發(fā)環(huán)境下才會使用module.hot的API,實現(xiàn)當reducer發(fā)生變化時的頁面渲染更新通知,注意其中的hydrate方法,這是v16版本的一個專門為服務(wù)端渲染新增的API方法,它在render方法的基礎(chǔ)上實現(xiàn)了對服務(wù)端渲染內(nèi)容的最大可能重用,實現(xiàn)了靜態(tài)DOM到動態(tài)NODES的過程。實質(zhì)是代替了v15版本下判斷checksum標記的過程,使得重用的過程更加高效優(yōu)雅。
const renderApp=()=>{ let application=createApp({store,history}); hydrate(application,document.getElementById('root')); } window.main = () => { Loadable.preloadReady().then(() => { renderApp() }); }; if(process.env.NODE_ENV==='development'){ if(module.hot){ module.hot.accept('./store/reducers/index.js',()=>{ let newReducer=require('./store/reducers/index.js'); store.replaceReducer(newReducer) }) module.hot.accept('./app/index.js',()=>{ let {createApp}=require('./app/index.js'); let newReducer=require('./store/reducers/index.js'); store.replaceReducer(newReducer) let application=createApp({store,history}); hydrate(application,document.getElementById('root')); }) } }
注意window.main這個函數(shù)的定義,結(jié)合index.ejs可以知道這個函數(shù)是所有腳本加載完成后才觸發(fā),里面用的是react-loadable的寫法,用于頁面的懶加載,關(guān)于頁面分別打包的寫法要結(jié)合路由設(shè)置來講解,這里有個大致印象即可。需要注意的是app這個文件下暴露出的三個方法是在瀏覽器端和服務(wù)器端通用的,接下來主要就是說這部分的思路。
路由處理
接下來看以下src/app目錄下的文件,index.js暴露了三個方法,這里面涉及的三個方法在服務(wù)端和瀏覽器端開發(fā)都會用到,這一部分主要講其下的router文件里面的代碼思路和createApp.js文件對路由的處理,這里是實現(xiàn)兩端路由相互打通的關(guān)鍵點。
router文件夾下的routes.js是路由配置文件,將各個頁面下的路由配置都引進來,合成一個配置數(shù)組,可以通過這個配置來靈活控制頁面上下線。同目錄下的index.js是RouterV4的標準寫法,通過遍歷配置數(shù)組的方式傳入路由配置,ConnectRouter是用于合并Router的一個組件,注意到history要作為參數(shù)傳入,需要在createApp.js文件里做單獨的處理。先大致看一下Route組件中的幾個配置項,值得注意的是其中的thunk屬性,這是實現(xiàn)后端獲取數(shù)據(jù)后渲染的關(guān)鍵一步,正是這個屬性實現(xiàn)了類似Next里面的組件提前獲取數(shù)據(jù)的生命周期鉤子,其余的屬性都可以在相關(guān)React-router文檔中找到說明,這里不在贅述。
import routesConfig from './routes'; const Routers=({history})=>() export default Routers; { routesConfig.map(route=>()) }
查看app目錄下的createApp.js里面的代碼可以發(fā)現(xiàn),本框架是針對不同的工作環(huán)境做了不同的處理,只有在生產(chǎn)環(huán)境下才利用Loadable.Capture方法實現(xiàn)了懶加載,動態(tài)引入不同頁面對應(yīng)的打包之后的js文件。到這里還要看一下組件里面的路由配置文件的寫法,以home頁面下的index.js為例。注意/* webpackChunkName: 'Home' */這串字符,實質(zhì)是指定了打包后此頁面對應(yīng)的js文件名,所以針對不同的頁面,這個注釋也需要修改,避免打包到一起。loading這個配置項只會在開發(fā)環(huán)境生效,當頁面加載未完成前顯示,這個實際項目開發(fā)如果不需要可以刪除此組件。
import {homeThunk} from '../../store/actions/thunk'; const LoadableHome = Loadable({ loader: () =>import(/* webpackChunkName: 'Home' */'./containers/homeContainer.js'), loading: Loading, }); const HomeRouter = { path: '/', exact: true, component: LoadableHome, thunk: homeThunk // 服務(wù)端渲染會開啟并執(zhí)行這個action,用于獲取頁面渲染所需數(shù)據(jù) } export default HomeRouter
這里多說一句,有時我們要改造的項目的頁面文件里有從window.location里面獲取參數(shù)的代碼,改造成服務(wù)端渲染時要全部去掉,或者是要在render之后的生命周期中使用。并且頁面級別組件都已經(jīng)注入了相關(guān)路由信息,可以通過this.props.location來獲取URL里面的參數(shù)。本項目用的是BrowserRouter,如果用HashRouter則包含參數(shù)可能略有不同,根據(jù)實際情況取用。
根據(jù)React16的服務(wù)端渲染的API介紹:
服務(wù)端渲染
這里就不會涉及到koa2的一些基礎(chǔ)知識,如果對koa2框架不熟悉可以參考我的另外一篇博文。這里是看server文件夾下都是服務(wù)端的代碼。首先是簡潔的app.js用于保證每次連接都返回的是一個新的服務(wù)器端實例,這對于單線程的js語言是很關(guān)鍵的思路。需要重點介紹的就是clientRouter.js這個文件,結(jié)合/src/app/configureStore.js這個文件共同理解服務(wù)端渲染的數(shù)據(jù)獲取流程和React的渲染機制。
/*configureStore.js*/ import {createStore, applyMiddleware,compose} from "redux"; import thunkMiddleware from "redux-thunk"; import createHistory from 'history/createMemoryHistory'; import { routerReducer, routerMiddleware } from 'react-router-redux' import rootReducer from '../store/reducers/index.js'; const routerReducers=routerMiddleware(createHistory());//路由 const composeEnhancers = process.env.NODE_ENV=='development'?window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__ : compose; const middleware=[thunkMiddleware,routerReducers]; //把路由注入到reducer,可以從reducer中直接獲取路由信息 let configureStore=(initialState)=>createStore(rootReducer,initialState,composeEnhancers(applyMiddleware(...middleware))); export default configureStore;
window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__這個變量是瀏覽器里面的Redux的開發(fā)者工具,開發(fā)React-redux應(yīng)用時建議安裝,否則會有報錯提示。這里面大部分都是redux-thunk的示例代碼,關(guān)于這部分如果看不懂建議看一下redux-thunk的官方文檔,這里要注意的是configureStore這個方法要傳入的initialState參數(shù),這個渲染的具體思路是:在服務(wù)端判斷路由的thunk方法,如果存在則需要執(zhí)行這個獲取數(shù)據(jù)邏輯,這是個阻塞過程,可以當作同步,獲取后放到全局State中,在前端輸出的HTML中注入window.__INITIAL_STATE__這個全局變量,當html載入完畢后,這個變量賦值已有數(shù)據(jù)的全局State作為initState提供給react應(yīng)用,然后瀏覽器端的js加載完畢后會通過復(fù)用頁面上已有的dom和初始的initState作為開始,合并到render后的生命周期中,從而在componentDidMount中已經(jīng)可以從this.props中獲取渲染所需數(shù)據(jù)。
但還要考慮到頁面切換也有可能在前端執(zhí)行跳轉(zhuǎn),此時作為React的應(yīng)用不會觸發(fā)對后端的請求,因此在componentDidMount這個生命周期里并沒有獲取數(shù)據(jù),為了解決這個問題,我建議在這個生命周期中都調(diào)用props中傳來的action觸發(fā)函數(shù),但在action內(nèi)部進行一層邏輯判斷,避免重復(fù)的請求,實際項目中請求數(shù)據(jù)往往會有個標識性ID,就可以將這個ID存入store中,然后就可以進行一次對比校驗來提前返回,避免重復(fù)發(fā)送ajax請求,具體可看store/actions/home.js`中的邏輯處理。
import {ADD,GET_HOME_INFO} from '../constants' export const add=(count)=>({type: ADD, count,}) export const getHomeInfo=(sendId=1)=>async(dispatch,getState)=>{ let {name,age,id}=getState().HomeReducer.homeInfo; if (id === sendId) { return //是通過對請求id和已有數(shù)據(jù)的標識性id進行對比校驗,避免重復(fù)獲取數(shù)據(jù)。 } console.log('footer'.includes('foo')) await new Promise(resolve=>{ let homeInfo={name:'wd2010',age:'25',id:sendId} console.log('-----------請求getHomeInfo') setTimeout(()=>resolve(homeInfo),1000) }).then(homeInfo=>{ dispatch({type:GET_HOME_INFO,data:{homeInfo}}) }) }
注意這里的async/await寫法,這里涉及到服務(wù)端koa2使用這個來做數(shù)據(jù)請求,因此需要統(tǒng)一返回async函數(shù),這塊不熟的同學(xué)建議看下ES7的知識,主要是async如何配合Promise實現(xiàn)異步流程改造,并且如果涉及koa2的服務(wù)端工作,對async函數(shù)用的更多,這也是本項目要求Node版本為8.x以上的原因,從8開始就可以直接用這兩個關(guān)鍵字。
不過到具體項目中,往往會涉及到一些服務(wù)端參數(shù)的注入問題,但這塊根據(jù)不同項目需求差異很大,并且不屬于這個React服務(wù)端改造的一部分,沒法統(tǒng)一分享,如果真是公司項目要用到對這塊有需求咨詢可以打賞后加我微信討論。
以Home頁面為例的渲染流程
為了方便大家理解,我以一個頁面為例整理了一下數(shù)據(jù)流的整體過程,看一下思路:
服務(wù)端直出HTML
基本的流程已經(jīng)介紹結(jié)束,至于一些Reducer的函數(shù)式寫法,還有actions的位置都是參考網(wǎng)上的一些分析來組織的,具體見仁見智,這個只要符合自己的理解,并且有助于團隊開發(fā)就好。如果您符合我在文章一開始設(shè)定的讀者背景,相信本文的講述足夠您點亮自己的服務(wù)端渲染技術(shù)點啦。如果對React了解偏少也沒關(guān)系,可以參考這里來補充一些React的基礎(chǔ)知識
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。