React项目的服务端渲染改造(koa2+webpack3.11)
选型思路:实现服务端渲染,想用React最新的版本,并且不对现有的写法做大的改动,如果一开始就打算服务端渲染,建议直接用
NEXT
框架来写
项目地址:https://github.com/wlx200510/react_koa_ssr
脚手架选型:webpack3.11.0 + react Router4 + Redux + koa2 + React16 + Node8.x
主要心得:对React
的相关知识更加熟悉,成功拓展自己的技术领域,对服务端技术在实际项目上有所积累
注意点:使用框架前一定确认当前webpack版本为3.x Node为8.x以上,读者最好用React在3个月以上,并有实际React项目经验
项目目录介绍
|
|
项目的构建思路
- 本地开发使用webpack-dev-server,实现热更新,基本流程跟之前react开发类似,仍是浏览器端渲染,因此在编写代码时要考虑到一套逻辑,两种渲染环境的问题。
- 当前端页面渲染完成后,其Router跳转将不会对服务端进行请求,从而减轻服务端压力,从而页面的进入方式也是两种,还要考虑两种渲染环境下路由同构的问题。
- 生产环境要使用koa做后端服务器,实现按需加载,在服务端获取数据,并渲染出整个HTML,利用React16最新的能力来合并整个状态树,实现服务端渲染。
本地开发介绍
查看本地开发主要涉及的文件是src
目录下的index.js
文件,判断当前的运行环境,只有在开发环境下才会使用module.hot的API,实现当reducer发生变化时的页面渲染更新通知,注意其中的hydrate
方法,这是v16版本的一个专门为服务端渲染新增的API
方法,它在render方法的基础上实现了对服务端渲染内容的最大可能重用,实现了静态DOM
到动态NODES
的过程。实质是代替了v15版本下判断checksum标记的过程,使得重用的过程更加高效优雅。
|
|
注意window.main
这个函数的定义,结合index.ejs
可以知道这个函数是所有脚本加载完成后才触发,里面用的是react-loadable
的写法,用于页面的懒加载,关于页面分别打包的写法要结合路由设置来讲解,这里有个大致印象即可。需要注意的是app这个文件下暴露出的三个方法是在浏览器端和服务器端通用的,接下来主要就是说这部分的思路。
路由处理
接下来看以下src/app
目录下的文件,index.js
暴露了三个方法,这里面涉及的三个方法在服务端和浏览器端开发都会用到,这一部分主要讲其下的router
文件里面的代码思路和createApp.js
文件对路由的处理,这里是实现两端路由相互打通的关键点。
router
文件夹下的routes.js是路由配置文件,将各个页面下的路由配置都引进来,合成一个配置数组,可以通过这个配置来灵活控制页面上下线。同目录下的index.js
是RouterV4
的标准写法,通过遍历配置数组的方式传入路由配置,ConnectRouter
是用于合并Router
的一个组件,注意到history
要作为参数传入,需要在createApp.js
文件里做单独的处理。先大致看一下Route组件中的几个配置项,值得注意的是其中的thunk
属性,这是实现后端获取数据后渲染的关键一步,正是这个属性实现了类似Next
里面的组件提前获取数据的生命周期钩子,其余的属性都可以在相关React-router文档中找到说明,这里不在赘述。
|
|
查看app
目录下的createApp.js
里面的代码可以发现,本框架是针对不同的工作环境做了不同的处理,只有在生产环境下才利用Loadable.Capture方法实现了懒加载,动态引入不同页面对应的打包之后的js文件。到这里还要看一下组件里面的路由配置文件的写法,以home
页面下的index.js
为例。注意/* webpackChunkName: 'Home' */
这串字符,实质是指定了打包后此页面对应的js文件名,所以针对不同的页面,这个注释也需要修改,避免打包到一起。loading
这个配置项只会在开发环境生效,当页面加载未完成前显示,这个实际项目开发如果不需要可以删除此组件。
|
|
这里多说一句,有时我们要改造的项目的页面文件里有从window.location
里面获取参数的代码,改造成服务端渲染时要全部去掉,或者是要在render之后的生命周期中使用。并且页面级别组件都已经注入了相关路由信息,可以通过this.props.location
来获取URL里面的参数。本项目用的是BrowserRouter
,如果用HashRouter
则包含参数可能略有不同,根据实际情况取用。
根据React16的服务端渲染的API介绍:
浏览器端使用的注入ConnectedRouter
中的history
为:import createHistory from 'history/createBrowserHistory'
服务器端使用的history
为import createHistory from 'history/createMemoryHistory'
服务端渲染
这里就不会涉及到koa2
的一些基础知识,如果对koa2
框架不熟悉可以参考我的另外一篇博文。这里是看server文件夹下都是服务端的代码。首先是简洁的app.js
用于保证每次连接都返回的是一个新的服务器端实例,这对于单线程的js语言是很关键的思路。需要重点介绍的就是clientRouter.js
这个文件,结合/src/app/configureStore.js
这个文件共同理解服务端渲染的数据获取流程和React的渲染机制。
|
|
window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__
这个变量是浏览器里面的Redux的开发者工具,开发React-redux应用时建议安装,否则会有报错提示。这里面大部分都是redux-thunk
的示例代码,关于这部分如果看不懂建议看一下redux-thunk
的官方文档,这里要注意的是configureStore
这个方法要传入的initialState
参数,这个渲染的具体思路是:在服务端判断路由的thunk方法,如果存在则需要执行这个获取数据逻辑,这是个阻塞过程,可以当作同步,获取后放到全局State中,在前端输出的HTML中注入window.__INITIAL_STATE__
这个全局变量,当html载入完毕后,这个变量赋值已有数据的全局State
作为initState
提供给react应用,然后浏览器端的js加载完毕后会通过复用页面上已有的dom和初始的initState
作为开始,合并到render后的生命周期中,从而在componentDidMount
中已经可以从this.props中获取渲染所需数据。
但还要考虑到页面切换也有可能在前端执行跳转,此时作为React的应用不会触发对后端的请求,因此在componentDidMount
这个生命周期里并没有获取数据,为了解决这个问题,我建议在这个生命周期中都调用props中传来的action
触发函数,但在action内部进行一层逻辑判断,避免重复的请求,实际项目中请求数据往往会有个标识性ID,就可以将这个ID存入store
中,然后就可以进行一次对比校验来提前返回,避免重复发送ajax
请求,具体可看store/actions/home.js`中的逻辑处理。
|
|
注意这里的async/await
写法,这里涉及到服务端koa2使用这个来做数据请求,因此需要统一返回async
函数,这块不熟的同学建议看下ES7的知识,主要是async
如何配合Promise
实现异步流程改造,并且如果涉及koa2的服务端工作,对async
函数用的更多,这也是本项目要求Node版本为8.x以上的原因,从8开始就可以直接用这两个关键字。
不过到具体项目中,往往会涉及到一些服务端参数的注入问题,但这块根据不同项目需求差异很大,并且不属于这个React服务端改造的一部分,没法统一分享,如果真是公司项目要用到对这块有需求咨询可以打赏后加我微信讨论。
以Home页面为例的渲染流程
为了方便大家理解,我以一个页面为例整理了一下数据流的整体过程,看一下思路:
- 服务端接收到请求,通过
/home
找到对应的路由配置 - 判断路由存在
thunk
方法,此时执行store/actions/thunk.js
里面的暴露出的函数 - 异步获取的数据会注入到全局state中,此时的dispatch分发其实并不生效
- 要输出的HTML代码中会将获取到数据后的全局state放到
window.__INITIAL_STATE__
这个全局变量中,作为initState window.__INITIAL_STATE__
将在react生命周期起作用前合并入全局state,此时react发现dom已经生成,不会再次触发render,并且数据状态得到同步
基本的流程已经介绍结束,至于一些Reducer
的函数式写法,还有actions的位置都是参考网上的一些分析来组织的,具体见仁见智,这个只要符合自己的理解,并且有助于团队开发就好。如果您符合我在文章一开始设定的读者背景,相信本文的讲述足够您点亮自己的服务端渲染技术点啦。如果对React了解偏少也没关系,可以参考这里来补充一些React的基础知识,也可以到我的博客学习交流。
如果这篇文章对您有帮助,或者用于您公司的项目发现一些问题,欢迎加我微信打赏后讨论并解决问题~。
打赏二维码:
打赏获取特权流程:
- 扫描下发二维码 加我为好友
- 发送打赏截图给我
- 拉你进React的服务端渲染讨论微信群
- 解答这个框架在实际应用中的问题