伴随着React协议的『妥协』(v16采用MIT),React为项目的主体,这个在短期内是不会改变的了,在平时使用过程中发现了如下这个问题:

在服务器渲染的时候,刷新页面会出现闪屏的现象(白屏一闪而过)

作为努力最求极致的我,是不能容忍的,而这一现象是半道出现的,也就是在添加按需加载之后。要说清楚这个问题,得从React的服务器渲染开始说起,(急于寻求问题解决方案的,可以直接去文章后半部分)

服务器渲染(SSR)基础原理

React的虚拟DOM是其可被用于服务端渲染的关键。其原理简单的来说就是首先每个ReactComponent 在虚拟DOM中完成渲染,然后React通过虚拟DOM来更新浏览器DOM中产生变化的那一部分。虚拟DOM作为内存中的DOM表现,为React在Node.js这类非浏览器环境下提供了可能。React可以从虚拟DOM中生成一个字符串,而不是更新真正的DOM,这使得我们可以在客户端和服务端使用同一个React Component。

基本设计理念

React 提供了两个可用于服务端渲染组件的函数:ReactDOMServer.renderToStringReactDOMServer.renderToStaticMarkup。 在设计用于服务端渲染的ReactComponent时需要有预见性,考虑以下方面:

  • 选取最优的渲染函数。

  • 如何支持组件的异步状态。

  • 如何将应用的初始化状态传递到客户端。

  • 哪些生命周期函数可以用于服务端的渲染。

  • 如何为应用提供同构路由支持。

  • 单例、实例以及上下文的用法。

渲染函数

render()

我们常见的render方法,用于浏览器渲染。

import ReactDOM from 'react-dom';
ReactDOM.render(
element,
container,
[callback]
)

renderToString()

ReactDOMServer.renderToString是两个服务端渲染函数中的一个,也是开发主要使用的一个函数,ReactDOM.render不同,该函数去掉了用于表示渲染位置的参数。取而代之,该函数只返回一个字符串,这是一个快速的同步(阻塞式)函数,非常快。

  • 用法:

const ReactDOMServer = require('react-dom/server');
ReactDOMServer.renderToString(element)
  • 例子:

const ReactDOMServer = require('react-dom/server');
const Hello = React.createClass({
render: function() {
  return <div>hello</div>;
}
});
const helloString = ReactDOMServer.renderToString(
React.createElement(Hello)
);
/*
输出结果大概为:
helloString = `
<div
data-reactid=".xxx"
data-react-checksum="-123456"
>
hello
</div>
`
*/

从上面这个例子,很容易发现,React为div注入了一些自定义属性,首先reactid,这是在浏览器环境下,React为了区分DOM节点,在需要更新的时候能够精确定位的标记。而后checksum这个属性仅仅存在与服务端,这个你可能没见过,或没留意,它的作用是拿服务端返回的String与已创建的DOM做校验,这就准许了React在客户端和服务端在结构上拥有相同的DOM结构,该属性只会添加在根节点元素上。拿到checksum大抵会做一些事情:

  • 检查第一个元素是否有data-react-checksum属性,如果有则通过ReactDOMServer.renderToString拿到前端的,通过adler32算法得到的值和data-react-checksum对比,如果一致则表示,无需渲染,否则重新渲染。

renderToStaticMarkup()

import ReactDOMServer from 'react-dom/server';
ReactDOMServer.renderToStaticMarkup(element)
// eg:
ReactDOMServer.renderToStaticMarkup(
React.createElement(
  Provider,
  { store },
  React.createElement(
    RouterContext,
    matchedData[1]
  )
)
);

这个函数和上面的那个函数大体相同,除却,它不会给节点添加任何额外的属性值,它的返回值是『干净』的,在某种情况下,可以节约空间使用。

选择

每个渲染函数都有自己的用途,所以你必须明确自己的需求,再去决定使用哪个渲染函数。当且仅当你不打算在客户端渲染这个React Component时,才应该选择使用ReactDOMServer.renderToStaticMarkup函数。下面有一些示例:

  • 生成HTML电子邮件

  • 通过HTML到PDF的转化来生成PDF

  • 组件测试

  • 等一些需要『纯』DOM的情况下

大多数情况下,我们都会选择使用ReactDOMServer.renderToString。这将准许React使用data-react-checksum在客户端迅速的初始化同一个React Component,因为React可以重用服务端提供的DOM,所以它可以跳过生成DOM节点以及把他们挂载到文档中这两个昂贵的进程,对于复杂些的站点,这样做就会显著的减少加载时间,用户可以更快的与站点进行交互。确保React Component能够在服务端和客户端准确的渲染出一致的结构是很重要的。如果data-react-checksum不匹配,React会舍弃服务端提供的DOM,然后生成新的DOM节点,并且将它们更新到文档中。此时,也就不具备服务端渲染带来的各种性能上的优势。这个错误会是下面这样的,如果你开了React dev模式:

Warning: React attempted to reuse markup in a container but the checksum was invalid. This generally means that you are using server rendering and the markup generated on the server was not what the client was expecting. React injected new markup to compensate which works but you have lost many of the benefits of server rendering. Instead, figure out why the markup being generated is different on the client or server:
(client) <noscript data-reacti
(server) <div data-reactid=".q

而解决这个问题,则是保证在服务器环境下能够渲染出与客户端一样的DOM结构,总结一下一般会是两种情况:

  • 异步/延迟加载

  • 存在随机逻辑

  • 最大的问题是,客户端和服务端的环境差异造就的问题,如document环境下的一些元素无法在服务端渲染等

总结

上面讲了些服务器渲染需要注意到的点,而一开始提到的按需加载刷新页面出现的闪屏问题还没有落实。以下:在异步加载的时候服务器渲染和客户端渲染的结果不同,即通过checksum校验失败,重新渲染,在这个重新渲染的时候,需要重新匹配路由,以上就会出现闪屏的情况。我们注意到,在使用react-router的时候,服务器中使用到了一个match函数

match({ history, routes: getRoutes(store), location: req.originalUrl }, (error, redirectLocation, renderProps) => {})

简单的说他匹配分析了客户端的路由,使得渲染过程中能够吻合到客户端的路由控制器。而如果客户端在首次出现需要重新渲染的时候,如果是动态路由(按需加载使用到的一项技术),就需要重新匹配渲染,这时候会出现短暂的白屏闪过。解决这个问题,只需要在客户端渲染之前先匹配路由,使用match。(关于match的详细介绍,参见官方文档)

// +redux
// 客户端在渲染前加上匹配路由函数match
match({ history, routes }, (error, redirectLocation, renderProps) => {
if (!error) {
  // 渲染
  ReactDOM.render(
    <Provider store={store}>{routes}</Provider>,
    document.getElementById('root')
  );
} else {
  console.error(error);
  // todo: 错误信息收集
}
});

以上,便可解决题中问题。

React Router 按需加载+服务器渲染的闪屏问题的更多相关文章

  1. vue router按需加载

    import Vue from 'vue' import Router from 'vue-router' Vue.use(Router); //按需加载,当渲染其他页面时才加载其组件,并缓存,减少首 ...

  2. 深入浅出的webpack4构建工具---webpack+vue+router 按需加载页面(十五)

    1. 为什么需要按需加载? 对于vue单页应用来讲,我们常见的做法把页面上所有的代码都打包到一个bundle.js文件内,但是随着项目越来越大,文件越来越多的情况下,那么bundle.js文件也会越来 ...

  3. react路由按需加载方法

    使用router4之后以前的按需加载方法require.ensure 是不好使了. 所以我们改用react-loadable插件做按需加载. 第一步: yarn add react-loadable ...

  4. 配置react / antd 按需加载 并且使用less(react v16)

    1.开启项目   并且执行 yarn eject 下载好我们需要的插件(babel-plugin-import   less  less-loader   antd  react-loadable   ...

  5. easyUI datagrid 刷新取消加载信息 自动刷新闪屏问题

    <style type="text/css"> /*-- 消除grid屏闪问题 --//*/ .datagrid-mask { opacity: 0; filter: ...

  6. React Router 4.0 + webpack 实现组件按需加载

    网上关于React Router 4.0的按需加载文章有很多,大致的思路都一样,但是其实具体实现起来却要根据自己的实际情况来定,这里主要介绍一下我的实现方式. 主要方式是通过Route组件的rende ...

  7. EasyDSS高性能流媒体服务器前端重构(五)- webpack + vue-router 开发单页面前端实现按需加载 - 副本

    为了让页面更快完成加载, 第一时间呈现给客户端, 也为了帮助客户端节省流量资源, 我们可以开启 vue-router 提供的按需加载功能, 让客户端打开页面时, 只自动加载必要的资源文件, 当客户端操 ...

  8. EasyDSS高性能RTMP、HLS(m3u8)、HTTP-FLV、RTSP流媒体服务器前端重构(五)- webpack + vue-router 开发单页面前端实现按需加载

    为了让页面更快完成加载, 第一时间呈现给客户端, 也为了帮助客户端节省流量资源, 我们可以开启 vue-router 提供的按需加载功能, 让客户端打开页面时, 只自动加载必要的资源文件, 当客户端操 ...

  9. react按需加载(getComponent优美写法),并指定输出模块名称解决缓存(getComponent与chunkFilename)

    react配合webpack进行按需加载的方法很简单,Route的component改为getComponent,组件用require.ensure的方式获取,并在webpack中配置chunkFil ...

随机推荐

  1. Eclipse知识

    http://www.runoob.com/eclipse/eclipse-create-jar-files.html Eclipse 生成jar包 打开 Jar 文件向导 Jar 文件向导可用于将项 ...

  2. php 数据访问练习:租房查询页面

    <html> <head> <title></title> <meta charset="UTF-8"/> <li ...

  3. MySQL(二)之服务管理与初始化文件修改和连接MySQL

    上一篇给大家介绍了怎么在linux和windows中安装mysql,本来是可以放在首页的,但是博客园说“安装配置类文件”不让放在首页.接下来给大家介绍一下在linux和windows下MySQL的一下 ...

  4. QT creator编程C++第一步,说“Hello world!”

    这个学期选了计算机学院的<数字图像处理>,正好和我的图像识别项目有所关联,老师说不能用MATLAB来做,这让我一个没学过C++的孩纸欲哭无泪. 只好求助计算机学院的大佬,自学C++. 大佬 ...

  5. 原创: rsync软件服务利用ansible实现一键化部署

    ---恢复内容开始--- 首先创建一个脚本文件 /server/tools/peizhi.sh cat  /server/tools/peizhi.sh cat >>/etc/rsyncd ...

  6. 近期学习的原生JS知识以及jQuery框架

    [正则表达式]1.正则表达式包括两部分: ① 定义正则表达式的规则 ② 定义正则表达式的模式(i/g/m)2.声明正则表达式: ① 字面声明 : var reg = /表达式规则/表达式模式 ② 使用 ...

  7. 总结各种排序算法【Java实现】

    一.插入类排序 1.直接插入排序 思想:将第i个插入到前i-1个中的适当位置 时间复杂度:T(n) = O(n²). 空间复杂度:S(n) = O(1). 稳定性:稳定排序. 如果碰见一个和插入元素相 ...

  8. chrome开发工具指南(十一)

    检查资源 使用 Application 面板的 Frames 窗格可以按框架组织资源. 您也可以在 Sources 面板中停用 Group by folder 选项,按框架查看资源. 要按网域和文件夹 ...

  9. 为什么Java 两个Integer 中1000==1000为false而100==100为true?

    详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt346 这是一个挺有意思的讨论话题. 如果你运行下面的代码 1 2 3 4 I ...

  10. WCF(二)三种通信模式

    WCF在通信过程中有三种模式:请求与答复.单向.双工通信 请求与答复模式 客户端发送请求,然后一直等待服务端的响应答复(异步调用除外),期间处于假死状态,直到服务端有了答复后才能继续执行其他程序 请求 ...