来源

笔者是一个 react 重度爱好者,在工作之余,也看了不少的 react 文章, 写了很多 react 项目 ,接下来笔者讨论一下 React 性能优化的主要方向和一些工作中的小技巧。送人玫瑰,手留余香,阅读的朋友可以给笔者点赞,关注一波 。陆续更新前端文章。
本文篇幅较长,将从 编译阶段 -> 路由阶段 -> 渲染阶段 -> 细节优化 -> 状态管理 -> 海量数据源,长列表渲染
方向分别加以探讨。

一 不能输在起跑线上,优化babel配置,webpack配置为项

1 真实项目中痛点

当我们用create-react-app或者webpack构建react工程的时候,有没有想过一个问题,我们的配置能否让我们的项目更快的构建速度,更小的项目体积,更简洁清晰的项目结构。随着我们的项目越做越大,项目依赖越来越多,项目结构越来越来复杂,项目体积就会越来越大,构建时间越来越长,久而久之就会成了一个又大又重的项目,所以说我们要学会适当的为项目‘减负’,让项目不能输在起跑线上。

2 一个老项目

拿我们之前接触过的一个react老项目为例。我们没有用dva,umi快速搭建react,而是用react老版本脚手架构建的,这对这种老的react项目,上述的问题都会存在,下面让我们一起来看看。
我们首先看一下项目结构。优化 - 图1
再看看构建时间。
为了方便大家看构建时间,我简单写了一个webpack,plugin ConsolePlugin ,记录了webpack在一次compilation所用的时间。

  1. const chalk = require('chalk') /* console 颜色 */
  2. var slog = require('single-line-log'); /* 单行打印 console */
  3. class ConsolePlugin {
  4. constructor(options){
  5. this.options = options
  6. }
  7. apply(compiler){
  8. /**
  9. * Monitor file change 记录当前改动文件
  10. */
  11. compiler.hooks.watchRun.tap('ConsolePlugin', (watching) => {
  12. const changeFiles = watching.watchFileSystem.watcher.mtimes
  13. for(let file in changeFiles){
  14. console.log(chalk.green('当前改动文件:'+ file))
  15. }
  16. })
  17. /**
  18. * before a new compilation is created. 开始 compilation 编译 。
  19. */
  20. compiler.hooks.compile.tap('ConsolePlugin',()=>{
  21. this.beginCompile()
  22. })
  23. /**
  24. * Executed when the compilation has completed. 一次 compilation 完成。
  25. */
  26. compiler.hooks.done.tap('ConsolePlugin',()=>{
  27. this.timer && clearInterval( this.timer )
  28. const endTime = new Date().getTime()
  29. const time = (endTime - this.starTime) / 1000
  30. console.log( chalk.yellow(' 编译完成') )
  31. console.log( chalk.yellow('编译用时:' + time + '秒' ) )
  32. })
  33. }
  34. beginCompile(){
  35. const lineSlog = slog.stdout
  36. let text = '开始编译:'
  37. /* 记录开始时间 */
  38. this.starTime = new Date().getTime()
  39. this.timer = setInterval(()=>{
  40. text += '█'
  41. lineSlog( chalk.green(text))
  42. },50)
  43. }
  44. }

构建时间如下:
优化 - 图2
打包后的体积:
优化 - 图3

3 翻新老项目

针对上面这个react老项目,我们开始针对性的优化。由于本文主要讲的是react,所以我们不把太多篇幅给webpack优化上。

① include 或 exclude 限制 loader 范围。

  1. {
  2. test: /\.jsx?$/,
  3. exclude: /node_modules/,
  4. include: path.resolve(__dirname, '../src'),
  5. use:['happypack/loader?id=babel']
  6. // loader: 'babel-loader'
  7. }

② happypack多进程编译

除了上述改动之外,在plugin中

  1. /* 多线程编译 */
  2. new HappyPack({
  3. id:'babel',
  4. loaders:['babel-loader?cacheDirectory=true']
  5. })

③缓存babel编译过的文件

  1. loaders:['babel-loader?cacheDirectory=true']

④tree Shaking 删除冗余代码

⑤按需加载,按需引入。

优化后项目结构
优化 - 图4
优化构建时间如下:
优化 - 图5
一次 compilation 时间 从23秒优化到了4.89秒
优化打包后的体积:
优化 - 图6
由此可见,如果我们的react是自己徒手搭建的,一些优化技巧显得格外重要。

关于类似antd UI库的瘦身思考

我们在做react项目的时候,会用到antd之类的ui库,值得思考的一件事是,如果我们只是用到了antd中的个别组件,比如<Button />,就要把整个样式库引进来,打包就会发现,体积因为引入了整个样式大了很多。我们可以通过.babelrc实现按需引入。
瘦身前
优化 - 图7
.babelrc 增加对 antd 样式按需引入。

  1. ["import", {
  2. "libraryName":
  3. "antd",
  4. "libraryDirectory": "es",
  5. "style": true
  6. }]

瘦身后
优化 - 图8

总结

如果想要优化react项目,从构建开始是必不可少的。我们要重视从构建到打包上线的每一个环节。

二 路由懒加载,路由监听器

react路由懒加载,是笔者看完dva源码中的 dynamic异步加载组件总结出来的,针对大型项目有很多页面,在配置路由的时候,如果没有对路由进行处理,一次性会加载大量路由,这对页面初始化很不友好,会延长页面初始化时间,所以我们想着用asyncRouter来按需加载页面路由。

传统路由

如果我们没有用umi等框架,需要手动配置路由的时候,也许路由会这样配置。

  1. <Switch>
  2. <Route path={'/index'} component={Index} ></Route>
  3. <Route path={'/list'} component={List} ></Route>
  4. <Route path={'/detail'} component={ Detail } ></Route>
  5. <Redirect from='/*' to='/index' />
  6. </Switch>

或者用list保存路由信息,方便在进行路由拦截,或者配置路由菜单等。

  1. const router = [
  2. {
  3. 'path': '/index',
  4. 'component': Index
  5. },
  6. {
  7. 'path': '/list'',
  8. 'component': List
  9. },
  10. {
  11. 'path': '/detail',
  12. 'component': Detail
  13. },
  14. ]

asyncRouter懒加载路由,并实现路由监听

我们今天讲的这种react路由懒加载是基于import 函数路由懒加载, 众所周知 ,import 执行会返回一个Promise作为异步加载的手段。我们可以利用这点来实现react异步加载路由
好的一言不合上代码。。。
代码

  1. const routerObserveQueue = [] /* 存放路由卫视钩子 */
  2. /* 懒加载路由卫士钩子 */
  3. export const RouterHooks = {
  4. /* 路由组件加载之前 */
  5. beforeRouterComponentLoad: function(callback) {
  6. routerObserveQueue.push({
  7. type: 'before',
  8. callback
  9. })
  10. },
  11. /* 路由组件加载之后 */
  12. afterRouterComponentDidLoaded(callback) {
  13. routerObserveQueue.push({
  14. type: 'after',
  15. callback
  16. })
  17. }
  18. }
  19. /* 路由懒加载HOC */
  20. export default function AsyncRouter(loadRouter) {
  21. return class Content extends React.Component {
  22. constructor(props) {
  23. super(props)
  24. /* 触发每个路由加载之前钩子函数 */
  25. this.dispatchRouterQueue('before')
  26. }
  27. state = {Component: null}
  28. dispatchRouterQueue(type) {
  29. const {history} = this.props
  30. routerObserveQueue.forEach(item => {
  31. if (item.type === type) item.callback(history)
  32. })
  33. }
  34. componentDidMount() {
  35. if (this.state.Component) return
  36. loadRouter()
  37. .then(module => module.default)
  38. .then(Component => this.setState({Component},
  39. () => {
  40. /* 触发每个路由加载之后钩子函数 */
  41. this.dispatchRouterQueue('after')
  42. }))
  43. }
  44. render() {
  45. const {Component} = this.state
  46. return Component ? <Component {
  47. ...this.props
  48. }
  49. /> : null
  50. }
  51. }
  52. }

asyncRouter实际就是一个高级组件,将()=>import()作为加载函数传进来,然后当外部Route加载当前组件的时候,在componentDidMount生命周期函数,加载真实的组件,并渲染组件,我们还可以写针对路由懒加载状态定制属于自己的路由监听器beforeRouterComponentLoadafterRouterComponentDidLoaded,类似vuewatch $route 功能。接下来我们看看如何使用。
使用

  1. import AsyncRouter ,{ RouterHooks } from './asyncRouter.js'
  2. const { beforeRouterComponentLoad} = RouterHooks
  3. const Index = AsyncRouter(()=>import('../src/page/home/index'))
  4. const List = AsyncRouter(()=>import('../src/page/list'))
  5. const Detail = AsyncRouter(()=>import('../src/page/detail'))
  6. const index = () => {
  7. useEffect(()=>{
  8. /* 增加监听函数 */
  9. beforeRouterComponentLoad((history)=>{
  10. console.log('当前激活的路由是',history.location.pathname)
  11. })
  12. },[])
  13. return <div >
  14. <div >
  15. <Router >
  16. <Meuns/>
  17. <Switch>
  18. <Route path={'/index'} component={Index} ></Route>
  19. <Route path={'/list'} component={List} ></Route>
  20. <Route path={'/detail'} component={ Detail } ></Route>
  21. <Redirect from='/*' to='/index' />
  22. </Switch>
  23. </Router>
  24. </div>
  25. </div>
  26. }

效果
image.gif
这样一来,我们既做到了路由的懒加载,又弥补了react-router没有监听当前路由变化的监听函数的缺陷。

三 受控性组件颗粒化 ,独立请求服务渲染单元

可控性组件颗粒化,独立请求服务渲染单元是笔者在实际工作总结出来的经验。目的就是避免因自身的渲染更新或是副作用带来的全局重新渲染。

1 颗粒化控制可控性组件

可控性组件和非可控性的区别就是dom元素值是否与受到react数据状态state控制。一旦由react的state控制数据状态,比如input输入框的值,就会造成这样一个场景,为了使input值实时变化,会不断setState,就会不断触发render函数,如果父组件内容简单还好,如果父组件比较复杂,会造成牵一发动全身,如果其他的子组件中componentWillReceiveProps这种带有副作用的钩子,那么引发的蝴蝶效应不敢想象。比如如下demo

  1. class index extends React.Component<any,any>{
  2. constructor(props){
  3. super(props)
  4. this.state={
  5. inputValue:''
  6. }
  7. }
  8. handerChange=(e)=> this.setState({ inputValue:e.target.value })
  9. render(){
  10. const { inputValue } = this.state
  11. return <div>
  12. { /* 我们增加三个子组件 */ }
  13. <ComA />
  14. <ComB />
  15. <ComC />
  16. <div className="box" >
  17. <Input value={inputValue} onChange={ (e)=> this.handerChange(e) } />
  18. </div>
  19. {/* 我们首先来一个列表循环 */}
  20. {
  21. new Array(10).fill(0).map((item,index)=>{
  22. console.log('列表循环了' )
  23. return <div key={index} >{item}</div>
  24. })
  25. }
  26. {
  27. /* 这里可能是更复杂的结构 */
  28. /* ------------------ */
  29. }
  30. </div>
  31. }
  32. }

组件A

  1. function index(){
  2. console.log('组件A渲染')
  3. return <div>我是组件A</div>
  4. }

组件B,有一个componentWillReceiveProps钩子

  1. class Index extends React.Component{
  2. constructor(props){
  3. super(props)
  4. }
  5. componentWillReceiveProps(){
  6. console.log('componentWillReceiveProps执行')
  7. /* 可能做一些骚操作 wu lian */
  8. }
  9. render(){
  10. console.log('组件B渲染')
  11. return <div>
  12. 我是组件B
  13. </div>
  14. }
  15. }

组件C有一个列表循环

  1. class Index extends React.Component{
  2. constructor(props){
  3. super(props)
  4. }
  5. render(){
  6. console.log('组件c渲染')
  7. return <div>
  8. 我是组件c
  9. {
  10. new Array(10).fill(0).map((item,index)=>{
  11. console.log('组件C列表循环了' )
  12. return <div key={index} >{item}</div>
  13. })
  14. }
  15. </div>
  16. }
  17. }

效果
image.gif
当我们在input输入内容的时候。就会造成如上的现象,所有的不该重新更新的地方,全部重新执行了一遍,这无疑是巨大的性能损耗。这个一个setState触发带来的一股巨大的由此组件到子组件可能更深的更新流,带来的副作用是不可估量的。所以我们可以思考一下,是否将这种受控性组件颗粒化,让自己更新 -> 渲染过程由自身调度。
说干就干,我们对上面的input表单单独颗粒化处理。

  1. const ComponentInput = memo(function({ notifyFatherChange }:any){
  2. const [ inputValue , setInputValue ] = useState('')
  3. const handerChange = useMemo(() => (e) => {
  4. setInputValue(e.target.value)
  5. notifyFatherChange && notifyFatherChange(e.target.value)
  6. },[])
  7. return <Input value={inputValue} onChange={ handerChange } />
  8. })

此时的组件更新由组件单元自行控制,不需要父组件的更新,所以不需要父组件设置独立state保留状态。只需要绑定到this上即可。不是所有状态都应该放在组件的 state 中. 例如缓存数据。如果需要组件响应它的变动, 或者需要渲染到视图中的数据才应该放到 state 中。这样可以避免不必要的数据变动导致组件重新渲染.

  1. class index extends React.Component<any,any>{
  2. formData :any = {}
  3. render(){
  4. return <div>
  5. { /* 我们增加三个子组件 */ }
  6. <ComA />
  7. <ComB />
  8. <ComC />
  9. <div className="box" >
  10. <ComponentInput notifyFatherChange={ (value)=>{ this.formData.inputValue = value } } />
  11. <Button onClick={()=> console.log(this.formData)} >打印数据</Button>
  12. </div>
  13. {/* 我们首先来一个列表循环 */}
  14. {
  15. new Array(10).fill(0).map((item,index)=>{
  16. console.log('列表循环了' )
  17. return <div key={index} >{item}</div>
  18. })
  19. }
  20. {
  21. /* 这里可能是更复杂的结构 */
  22. /* ------------------ */
  23. }
  24. </div>
  25. }
  26. }

效果
image.gif
这样除了当前组件外,其他地方没有收到任何渲染波动,达到了我们想要的目的。

2 建立独立的请求渲染单元

建立独立的请求渲染单元,直接理解就是,如果我们把页面,分为请求数据展示部分(通过调用后端接口,获取数据),和基础部分(不需要请求数据,已经直接写好的),对于一些逻辑交互不是很复杂的数据展示部分,我推荐用一种独立组件,独立请求数据,独立控制渲染的模式。至于为什么我们可以慢慢分析。
首先我们看一下传统的页面模式。
image.gif
页面有三个展示区域分别,做了三次请求,触发了三次setState,渲染三次页面,即使用Promise.all等方法,但是也不保证接下来交互中,会有部分展示区重新拉取数据的可能。一旦有一个区域重新拉取数据,另外两个区域也会说、受到牵连,这种效应是不可避免的,即便react有很好的ddiff算法去调协相同的节点,但是比如长列表等情况,循环在所难免。

  1. class Index extends React.Component{
  2. state :any={
  3. dataA:null,
  4. dataB:null,
  5. dataC:null
  6. }
  7. async componentDidMount(){
  8. /* 获取A区域数据 */
  9. const dataA = await getDataA()
  10. this.setState({ dataA })
  11. /* 获取B区域数据 */
  12. const dataB = await getDataB()
  13. this.setState({ dataB })
  14. /* 获取C区域数据 */
  15. const dataC = await getDataC()
  16. this.setState({ dataC })
  17. }
  18. render(){
  19. const { dataA , dataB , dataC } = this.state
  20. console.log(dataA,dataB,dataC)
  21. return <div>
  22. <div> { /* 用 dataA 数据做展示渲染 */ } </div>
  23. <div> { /* 用 dataB 数据做展示渲染 */ } </div>
  24. <div> { /* 用 dataC 数据做展示渲染 */ } </div>
  25. </div>
  26. }
  27. }

接下来我们,把每一部分抽取出来,形成独立的渲染单元,每个组件都独立数据请求到独立渲染。

  1. function ComponentA(){
  2. const [ dataA, setDataA ] = useState(null)
  3. useEffect(()=>{
  4. getDataA().then(res=> setDataA(res.data) )
  5. },[])
  6. return <div> { /* 用 dataA 数据做展示渲染 */ } </div>
  7. }
  8. function ComponentB(){
  9. const [ dataB, setDataB ] = useState(null)
  10. useEffect(()=>{
  11. getDataB().then(res=> setDataB(res.data) )
  12. },[])
  13. return <div> { /* 用 dataB 数据做展示渲染 */ } </div>
  14. }
  15. function ComponentC(){
  16. const [ dataC, setDataC ] = useState(null)
  17. useEffect(()=>{
  18. getDataC().then(res=> setDataC(res.data) )
  19. },[])
  20. return <div> { /* 用 dataC 数据做展示渲染 */ } </div>
  21. }
  22. function Index (){
  23. return <div>
  24. <ComponentA />
  25. <ComponentB />
  26. <ComponentC />
  27. </div>
  28. }

这样一来,彼此的数据更新都不会相互影响。
image.gif

总结

拆分需要单独调用后端接口的细小组件,建立独立的数据请求和渲染,这种依赖数据更新 -> 视图渲染的组件,能从整个体系中抽离出来 ,好处我总结有以下几个方面。
1 可以避免父组件的冗余渲染 ,react的数据驱动,依赖于 stateprops 的改变,改变state必然会对组件 render 函数调用,如果父组件中的子组件过于复杂,一个自组件的 state 改变,就会牵一发动全身,必然影响性能,所以如果把很多依赖请求的组件抽离出来,可以直接减少渲染次数。
2 可以优化组件自身性能,无论从class声明的有状态组件还是fun声明的无状态,都有一套自身优化机制,无论是用shouldupdate 还是用 hooksuseMemo useCallback ,都可以根据自身情况,定制符合场景的渲条 件,使得依赖数据请求组件形成自己一个小的,适合自身的渲染环境。
3 能够和redux ,以及redux衍生出来 redux-action , dva,更加契合的工作,用 connect 包裹的组件,就能通过制定好的契约,根据所需求的数据更新,而更新自身,而把这种模式用在这种小的,需要数据驱动的组件上,就会起到物尽其用的效果。

四 shouldComponentUpdate ,PureComponent 和 React.memo ,immetable.js 助力性能调优

在这里我们拿immetable.js为例,讲最传统的限制更新方法,第六部分将要将一些避免重新渲染的细节。

1 PureComponent 和 React.memo

React.PureComponentReact.Component 用法差不多 ,但React.PureComponent 通过props和state的浅对比来实现 shouldComponentUpate()。如果对象包含复杂的数据结构(比如对象和数组),他会浅比较,如果深层次的改变,是无法作出判断的,React.PureComponent 认为没有变化,而没有渲染试图。
如这个例子

  1. class Text extends React.PureComponent<any,any>{
  2. render(){
  3. console.log(this.props)
  4. return <div>hello,wrold</div>
  5. }
  6. }
  7. class Index extends React.Component<any,any>{
  8. state={
  9. data:{ a : 1 , b : 2 }
  10. }
  11. handerClick=()=>{
  12. const { data } = this.state
  13. data.a++
  14. this.setState({ data })
  15. }
  16. render(){
  17. const { data } = this.state
  18. return <div>
  19. <button onClick={ this.handerClick } >点击</button>
  20. <Text data={data} />
  21. </div>
  22. }
  23. }

效果
image.gif
我们点击按钮,发现 <Text /> 根本没有重新更新。这里虽然改了data但是只是改变了data下的属性,所以 PureComponent 进行浅比较不会update
想要解决这个问题实际也很容易。

  1. <Text data={{ ...data }} />

无论组件是否是 PureComponent,如果定义了 shouldComponentUpdate(),那么会调用它并以它的执行结果来判断是否 update。在组件未定义 shouldComponentUpdate() 的情况下,会判断该组件是否是 PureComponent,如果是的话,会对新旧 props、state 进行 shallowEqual 比较,一旦新旧不一致,会触发渲染更新。
react.memoPureComponent 功能类似 ,react.memo 作为第一个高阶组件,第二个参数 可以对props 进行比较 ,和shouldComponentUpdate不同的, 当第二个参数返回 true 的时候,证明props没有改变,不渲染组件,反之渲染组件。

2 shouldComponentUpdate

使用 shouldComponentUpdate()以让React知道当state或props的改变是否影响组件的重新render,默认返回ture,返回false时不会重新渲染更新,而且该方法并不会在初始化渲染或当使用 forceUpdate() 时被调用,通常一个shouldComponentUpdate 应用是这么写的。
控制状态

  1. shouldComponentUpdate(nextProps, nextState) {
  2. /* 当 state 中 data1 发生改变的时候,重新更新组件 */
  3. return nextState.data1 !== this.state.data1
  4. }

这个的意思就是 仅当statedata1 发生改变的时候,重新更新组件。 控制prop属性

  1. shouldComponentUpdate(nextProps, nextState) {
  2. /* 当 props 中 data2发生改变的时候,重新更新组件 */
  3. return nextProps.data2 !== this.props.data2
  4. }

这个的意思就是 仅当propsdata2 发生改变的时候,重新更新组件。

3 immetable.js

immetable.js 是Facebook 开发的一个js库,可以提高对象的比较性能,像之前所说的pureComponent 只能对对象进行浅比较,,对于对象的数据类型,却束手无策,所以我们可以用 immetable.js 配合 shouldComponentUpdate 或者 react.memo来使用。immutable
我们用react-redux来简单举一个例子,如下所示 数据都已经被 immetable.js处理。

  1. import { is } from 'immutable'
  2. const GoodItems = connect(state =>
  3. ({ GoodItems: filter(state.getIn(['Items', 'payload', 'list']), state.getIn(['customItems', 'payload', 'list'])) || Immutable.List(), })
  4. /* 此处省略很多代码~~~~~~ */
  5. )(memo(({ Items, dispatch, setSeivceId }) => {
  6. /* */
  7. }, (pre, next) => is(pre.Items, next.Items)))

通过 is 方法来判断,前后Items(对象数据类型)是否发生变化。

五 规范写法,合理处理细节问题

有的时候,我们在敲代码的时候,稍微注意一下,就能避免性能的开销。也许只是稍加改动,就能其他优化性能的效果。

①绑定事件尽量不要使用箭头函数

面临问题

众所周知,react更新来大部分情况来自于props的改变(被动渲染),和state改变(主动渲染)。当我们给未加任何更新限定条件子组件绑定事件的时候,或者是PureComponent 纯组件, 如果我们箭头函数使用的话。

  1. <ChildComponent handerClick={()=>{ console.log(666) }} />

每次渲染时都会创建一个新的事件处理器,这会导致 ChildComponent 每次都会被渲染。
即便我们用箭头函数绑定给dom元素。

  1. <div onClick={ ()=>{ console.log(777) } } >hello,world</div>

每次react合成事件事件的时候,也都会重新声明一个新事件。

解决问题

解决这个问题事件很简单,分为无状态组件和有状态组件。
有状态组件

  1. class index extends React.Component{
  2. handerClick=()=>{
  3. console.log(666)
  4. }
  5. handerClick1=()=>{
  6. console.log(777)
  7. }
  8. render(){
  9. return <div>
  10. <ChildComponent handerClick={ this.handerClick } />
  11. <div onClick={ this.handerClick1 } >hello,world</div>
  12. </div>
  13. }
  14. }

无状态组件

  1. function index(){
  2. const handerClick1 = useMemo(()=>()=>{
  3. console.log(777)
  4. },[]) /* [] 存在当前 handerClick1 的依赖项*/
  5. const handerClick = useCallback(()=>{ console.log(666) },[]) /* [] 存在当前 handerClick 的依赖项*/
  6. return <div>
  7. <ChildComponent handerClick={ handerClick } />
  8. <div onClick={ handerClick1 } >hello,world</div>
  9. </div>
  10. }

对于dom,如果我们需要传递参数。我们可以这么写。

  1. function index(){
  2. const handerClick1 = useMemo(()=>(event)=>{
  3. const mes = event.currentTarget.dataset.mes
  4. console.log(mes) /* hello,world */
  5. },[])
  6. return <div>
  7. <div data-mes={ 'hello,world' } onClick={ handerClick1 } >hello,world</div>
  8. </div>
  9. }

②循环正确使用key

无论是reactvue,正确使用key,目的就是在一次循环中,找到与新节点对应的老节点,复用节点,节省开销。想深入理解的同学可以看一下笔者的另外一篇文章 全面解析 vue3.0 diff算法 里面有对key详细说明。我们今天来看以下key正确用法,和错误用法。

1 错误用法

错误用法一:用index做key

  1. function index(){
  2. const list = [ { id:1 , name:'哈哈' } , { id:2, name:'嘿嘿' } ,{ id:3 , name:'嘻嘻' } ]
  3. return <div>
  4. <ul>
  5. { list.map((item,index)=><li key={index} >{ item.name }</li>) }
  6. </ul>
  7. </div>
  8. }

这种加key的性能,实际和不加key效果差不多,每次还是从头到尾diff。
错误用法二:用index拼接其他的字段

  1. function index(){
  2. const list = [ { id:1 , name:'哈哈' } , { id:2, name:'嘿嘿' } ,{ id:3 , name:'嘻嘻' } ]
  3. return <div>
  4. <ul>
  5. { list.map((item,index)=><li key={index + item.name } >{ item.name }</li>) }
  6. </ul>
  7. </div>
  8. }

如果有元素移动或者删除,那么就失去了一一对应关系,剩下的节点都不能有效复用。

2 正确用法

正确用法:用唯一id作为key

  1. function index(){
  2. const list = [ { id:1 , name:'哈哈' } , { id:2, name:'嘿嘿' } ,{ id:3 , name:'嘻嘻' } ]
  3. return <div>
  4. <ul>
  5. { list.map((item,index)=><li key={ item.id } >{ item.name }</li>) }
  6. </ul>
  7. </div>
  8. }

用唯一的健id作为key,能够做到有效复用元素节点。

③无状态组件hooks-useMemo 避免重复声明。

对于无状态组件,数据更新就等于函数上下文的重复执行。那么函数里面的变量,方法就会重新声明。比如如下情况。

  1. function Index(){
  2. const [ number , setNumber ] = useState(0)
  3. const handerClick1 = ()=>{
  4. /* 一些操作 */
  5. }
  6. const handerClick2 = ()=>{
  7. /* 一些操作 */
  8. }
  9. const handerClick3 = ()=>{
  10. /* 一些操作 */
  11. }
  12. return <div>
  13. <a onClick={ handerClick1 } >点我有惊喜1</a>
  14. <a onClick={ handerClick2 } >点我有惊喜2</a>
  15. <a onClick={ handerClick3 } >点我有惊喜3</a>
  16. <button onClick={ ()=> setNumber(number+1) } > 点击 { number } </button>
  17. </div>
  18. }

每次点击button的时候,都会执行Index函数。handerClick1 , handerClick2,handerClick3都会重新声明。为了避免这个情况的发生,我们可以用 useMemo 做缓存,我们可以改成如下。

  1. function Index(){
  2. const [ number , setNumber ] = useState(0)
  3. const [ handerClick1 , handerClick2 ,handerClick3] = useMemo(()=>{
  4. const fn1 = ()=>{
  5. /* 一些操作 */
  6. }
  7. const fn2 = ()=>{
  8. /* 一些操作 */
  9. }
  10. const fn3= ()=>{
  11. /* 一些操作 */
  12. }
  13. return [fn1 , fn2 ,fn3]
  14. },[]) /* 只有当数据里面的依赖项,发生改变的时候,才会重新声明函数。*/
  15. return <div>
  16. <a onClick={ handerClick1 } >点我有惊喜1</a>
  17. <a onClick={ handerClick2 } >点我有惊喜2</a>
  18. <a onClick={ handerClick3 } >点我有惊喜3</a>
  19. <button onClick={ ()=> setNumber(number+1) } > 点击 { number } </button>
  20. </div>
  21. }

如下改变之后,handerClick1 , handerClick2,handerClick3 会被缓存下来。

④懒加载 Suspense 和 lazy

Suspenselazy 可以实现 dynamic import 懒加载效果,原理和上述的路由懒加载差不多。在 React 中的使用方法是在 Suspense 组件中使用 <LazyComponent>组件。

  1. const LazyComponent = React.lazy(() => import('./LazyComponent'));
  2. function demo () {
  3. return (
  4. <div>
  5. <Suspense fallback={<div>Loading...</div>}>
  6. <LazyComponent />
  7. </Suspense>
  8. </div>
  9. )
  10. }

LazyComponent 是通过懒加载加载进来的,所以渲染页面的时候可能会有延迟,但使用了 Suspense 之后,在加载状态下,可以用<div>Loading...</div>作为loading效果。
Suspense 可以包裹多个懒加载组件。

  1. <Suspense fallback={<div>Loading...</div>}>
  2. <LazyComponent />
  3. <LazyComponent1 />
  4. </Suspense>

六 多种方式避免重复渲染

避免重复渲染,是react性能优化的重要方向。如果想尽心尽力处理好react项目每一个细节,那么就要从每一行代码开始,从每一组件开始。正所谓不积硅步无以至千里。

① 学会使用的批量更新

批量更新

这次讲的批量更新的概念,实际主要是针对无状态组件和hooksuseState,和 class有状态组件中的this.setState,两种方法已经做了批量更新的处理。比如如下例子
一次更新中

  1. class index extends React.Component{
  2. constructor(prop){
  3. super(prop)
  4. this.state = {
  5. a:1,
  6. b:2,
  7. c:3,
  8. }
  9. }
  10. handerClick=()=>{
  11. const { a,b,c } :any = this.state
  12. this.setState({ a:a+1 })
  13. this.setState({ b:b+1 })
  14. this.setState({ c:c+1 })
  15. }
  16. render= () => <div onClick={this.handerClick} />
  17. }

点击事件发生之后,会触发三次 setState,但是不会渲染三次,因为有一个批量更新batchUpdate批量更新的概念。三次setState最后被合成类似如下样子

  1. this.setState({
  2. a:a+1 ,
  3. b:b+1 ,
  4. c:c+1
  5. })

无状态组件中

  1. const [ a , setA ] = useState(1)
  2. const [ b , setB ] = useState({})
  3. const [ c , setC ] = useState(1)
  4. const handerClick = () => {
  5. setB( { ...b } )
  6. setC( c+1 )
  7. setA( a+1 )
  8. }

批量更新失效

当我们针对上述两种情况加以如下处理之后。

  1. handerClick=()=>{
  2. setTimeout(() => {
  3. this.setState({ a:a+1 })
  4. this.setState({ b:b+1 })
  5. this.setState({ c:c+1 })
  6. }, 0)
  7. }
  1. const handerClick = () => {
  2. Promise.resolve().then(()=>{
  3. setB( { ...b } )
  4. setC( c+1 )
  5. setA( a+1 )
  6. })
  7. }

我们会发现,上述两种情况 ,组件都更新渲染了三次 ,此时的批量更新失效了。这种情况在react-hooks中也普遍存在,这种情况甚至在hooks中更加明显,因为我们都知道hooks中每个useState保存了一个状态,并不是让class声明组件中,可以通过this.state统一协调状态,再一次异步函数中,比如说一次ajax请求后,想通过多个useState改变状态,会造成多次渲染页面,为了解决这个问题,我们可以手动批量更新。

手动批量更新

react-dom 中提供了unstable_batchedUpdates方法进行手动批量更新。这个api更契合react-hooks,我们可以这样做。

  1. const handerClick = () => {
  2. Promise.resolve().then(()=>{
  3. unstable_batchedUpdates(()=>{
  4. setB( { ...b } )
  5. setC( c+1 )
  6. setA( a+1 )
  7. })
  8. })
  9. }

这样三次更新,就会合并成一次。同样达到了批量更新的效果。

② 合并state

class类组件(有状态组件)

合并state这种,是一种我们在react项目开发中要养成的习惯。我看过有些同学的代码中可能会这么写(如下demo是模拟的情况,实际要比这复杂的多)。

  1. class Index extends React.Component<any , any>{
  2. state = {
  3. loading:false /* 用来模拟loading效果 */,
  4. list:[],
  5. }
  6. componentDidMount(){
  7. /* 模拟一个异步请求数据场景 */
  8. this.setState({ loading : true }) /* 开启loading效果 */
  9. Promise.resolve().then(()=>{
  10. const list = [ { id:1 , name: 'xixi' } ,{ id:2 , name: 'haha' },{ id:3 , name: 'heihei' } ]
  11. this.setState({ loading : false },()=>{
  12. this.setState({
  13. list:list.map(item=>({
  14. ...item,
  15. name:item.name.toLocaleUpperCase()
  16. }))
  17. })
  18. })
  19. })
  20. }
  21. render(){
  22. const { list } = this.state
  23. return <div>{
  24. list.map(item=><div key={item.id} >{ item.name }</div>)
  25. }</div>
  26. }
  27. }

分别用两次this.state第一次解除loading状态,第二次格式化数据列表。这另两次更新完全没有必要,可以用一次setState更新完美解决。不这样做的原因是,对于像demo这样的简单结构还好,对于复杂的结构,一次更新可能都是宝贵的,所以我们应该学会去合并state。将上述demo这样修改。

  1. this.setState({
  2. loading : false,
  3. list:list.map(item=>({
  4. ...item,
  5. name:item.name.toLocaleUpperCase()
  6. }))
  7. })

函数组件(无状态组件)

对于无状态组件,我们可以通过一个useState保存多个状态,没有必要每一个状态都用一个useState
对于这样的情况。

  1. const [ a ,setA ] = useState(1)
  2. const [ b ,setB ] = useState(2)

我们完全可以一个state搞定。

  1. const [ numberState , setNumberState ] = useState({ a:1 , b :2})

但是要注意,如果我们的state已经成为 useEffect , useCallback , useMemo依赖项,请慎用如上方法。

③ useMemo React.memo隔离单元

react正常的更新流,就像利剑一下,从父组件项子组件穿透,为了避免这些重复的更新渲染,shouldComponentUpdate , React.memoapi也应运而生。但是有的情况下,多余的更新在所难免,比如如下这种情况。这种更新会由父组件 -> 子组件 传递下去。
image.gif

  1. function ChildrenComponent(){
  2. console.log(2222)
  3. return <div>hello,world</div>
  4. }
  5. function Index (){
  6. const [ list ] = useState([ { id:1 , name: 'xixi' } ,{ id:2 , name: 'haha' },{ id:3 , name: 'heihei' } ])
  7. const [ number , setNumber ] = useState(0)
  8. return <div>
  9. <span>{ number }</span>
  10. <button onClick={ ()=> setNumber(number + 1) } >点击</button>
  11. <ul>
  12. {
  13. list.map(item=>{
  14. console.log(1111)
  15. return <li key={ item.id } >{ item.name }</li>
  16. })
  17. }
  18. </ul>
  19. <ChildrenComponent />
  20. </div>
  21. }

效果
image.gif
针对这一现象,我们可以通过使用useMemo进行隔离,形成独立的渲染单元,每次更新上一个状态会被缓存,循环不会再执行,子组件也不会再次被渲染,我们可以这么做。

  1. function Index (){
  2. const [ list ] = useState([ { id:1 , name: 'xixi' } ,{ id:2 , name: 'haha' },{ id:3 , name: 'heihei' } ])
  3. const [ number , setNumber ] = useState(0)
  4. return <div>
  5. <span>{ number }</span>
  6. <button onClick={ ()=> setNumber(number + 1) } >点击</button>
  7. <ul>
  8. {
  9. useMemo(()=>(list.map(item=>{
  10. console.log(1111)
  11. return <li key={ item.id } >{ item.name }</li>
  12. })),[ list ])
  13. }
  14. </ul>
  15. { useMemo(()=> <ChildrenComponent />,[]) }
  16. </div>
  17. }

有状态组件
class声明的组件中,没有像 useMemoAPI ,但是也并不等于束手无策,我们可以通过 react.memo 来阻拦来自组件本身的更新。我们可以写一个组件,来控制react 组件更新的方向。我们通过一个 <NotUpdate> 组件来阻断更新流。

  1. /* 控制更新 ,第二个参数可以作为组件更新的依赖 , 这里设置为 ()=> true 只渲染一次 */
  2. const NotUpdate = React.memo(({ children }:any)=> typeof children === 'function' ? children() : children ,()=>true)
  3. class Index extends React.Component<any,any>{
  4. constructor(prop){
  5. super(prop)
  6. this.state = {
  7. list: [ { id:1 , name: 'xixi' } ,{ id:2 , name: 'haha' },{ id:3 , name: 'heihei' } ],
  8. number:0,
  9. }
  10. }
  11. handerClick = ()=>{
  12. this.setState({ number:this.state.number + 1 })
  13. }
  14. render(){
  15. const { list }:any = this.state
  16. return <div>
  17. <button onClick={ this.handerClick } >点击</button>
  18. <NotUpdate>
  19. {()=>(<ul>
  20. {
  21. list.map(item=>{
  22. console.log(1111)
  23. return <li key={ item.id } >{ item.name }</li>
  24. })
  25. }
  26. </ul>)}
  27. </NotUpdate>
  28. <NotUpdate>
  29. <ChildrenComponent />
  30. </NotUpdate>
  31. </div>
  32. }
  33. }
  1. const NotUpdate = React.memo(({ children }:any)=> typeof children === 'function' ? children() : children ,()=>true)

image.gif
没错,用的就是 React.memo,生成了阻断更新的隔离单元,如果我们想要控制更新,可以对 React.memo 第二个参数入手, demo项目中完全阻断的更新。

④ ‘取缔’state,学会使用缓存。

这里的取缔state,并完全不使用state来管理数据,而是善于使用state,知道什么时候使用,怎么使用。react 并不像 vue 那样响应式数据流。在 vue中 有专门的dep做依赖收集,可以自动收集字符串模版的依赖项,只要没有引用的data数据, 通过 this.aaa = bbb ,在vue中是不会更新渲染的。因为 aaadep没有收集渲染watcher依赖项。在react中,我们触发this.setState 或者 useState,只会关心两次state值是否相同,来触发渲染,根本不会在乎jsx语法中是否真正的引入了正确的值。

没有更新作用的state

有状态组件中

  1. class Demo extends React.Component{
  2. state={ text:111 }
  3. componentDidMount(){
  4. const { a } = this.props
  5. /* 我们只是希望在初始化,用text记录 props中 a 的值 */
  6. this.setState({
  7. text:a
  8. })
  9. }
  10. render(){
  11. /* 没有引入text */
  12. return <div>{'hello,world'}</div>
  13. }
  14. }

如上例子中,render函数中并没有引入text ,我们只是希望在初始化的时候,用 text 记录 propsa 的值。我们却用 setState 触发了一次无用的更新。无状态组件中情况也一样存在,具体如下。
无状态组件中

  1. function Demo ({ a }){
  2. const [text , setText] = useState(111)
  3. useEffect(()=>{
  4. setText(a)
  5. },[])
  6. return <div>
  7. {'hello,world'}
  8. </div>
  9. }

改为缓存

有状态组件中
class声明组件中,我们可以直接把数据绑定给this上,来作为数据缓存。

  1. class Demo extends React.Component{
  2. text = 111
  3. componentDidMount(){
  4. const { a } = this.props
  5. /* 数据直接保存在text上 */
  6. this.text = a
  7. }
  8. render(){
  9. /* 没有引入text */
  10. return <div>{'hello,world'}</div>
  11. }
  12. }

无状态组件中
在无状态组件中, 我们不能往问this,但是我们可以用useRef来解决问题。

  1. function Demo ({ a }){
  2. const text = useRef(111)
  3. useEffect(()=>{
  4. text.current = a
  5. },[])
  6. return <div>
  7. {'hello,world'}
  8. </div>
  9. }

⑤ useCallback回调

useCallback 的真正目的还是在于缓存了每次渲染时 inline callback 的实例,这样方便配合上子组件的 shouldComponentUpdate 或者 React.memo 起到减少不必要的渲染的作用。对子组件的渲染限定来源与,对子组件props比较,但是如果对父组件的callback做比较,无状态组件每次渲染执行,都会形成新的callback ,是无法比较,所以需要对callback做一个 memoize 记忆功能,我们可以理解为useCallback就是 callback加了一个memoize。我们接着往下看👇👇👇。

  1. function demo (){
  2. const [ number , setNumber ] = useState(0)
  3. return <div>
  4. <DemoComponent handerChange={ ()=>{ setNumber(number+1) } } />
  5. </div>
  6. }

或着

  1. function demo (){
  2. const [ number , setNumber ] = useState(0)
  3. const handerChange = ()=>{
  4. setNumber(number+1)
  5. }
  6. return <div>
  7. <DemoComponent handerChange={ handerChange } />
  8. </div>
  9. }

无论是上述那种方式,pureComponentreact.memo 通过浅比较方式,只能判断每次更新都是新的callback,然后触发渲染更新。useCallback给加了一个记忆功能,告诉我们子组件,两次是相同的 callback无需重新更新页面。至于什么时候callback更改,就要取决于 useCallback 第二个参数。好的,将上述demo我们用 useCallback 重写。

  1. function demo (){
  2. const [ number , setNumber ] = useState(0)
  3. const handerChange = useCallback( ()=>{
  4. setNumber(number+1)
  5. },[])
  6. return <div>
  7. <DemoComponent handerChange={ handerChange } />
  8. </div>
  9. }

这样 pureComponentreact.memo 可以直接判断是callback没有改变,防止了不必要渲染。

七 中规中矩的使用状态管理

无论我们使用的是redux还是说 redux 衍生出来的 dva ,redux-saga等,或者是mobx,都要遵循一定’使用规则’,首先让我想到的是,什么时候用状态管理,怎么合理的应用状态管理,接下来我们来分析一下。

什么时候使用状态管理

要问我什么时候适合使用状态状态管理。我一定会这么分析,首先状态管理是为了解决什么问题,状态管理能够解决的问题主要分为两个方面,一 就是解决跨层级组件通信问题 。二 就是对一些全局公共状态的缓存。
我们那redux系列的状态管理为例子。
我见过又同学这么写的

滥用状态管理

  1. /* 和 store下面text模块的list列表,建立起依赖关系,list更新,组件重新渲染 */
  2. @connect((store)=>({ list:store.text.list }))
  3. class Text extends React.Component{
  4. constructor(prop){
  5. super(prop)
  6. }
  7. componentDidMount(){
  8. /* 初始化请求数据 */
  9. this.getList()
  10. }
  11. getList=()=>{
  12. const { dispatch } = this.props
  13. /* 获取数据 */
  14. dispatch({ type:'text/getDataList' })
  15. }
  16. render(){
  17. const { list } = this.props
  18. return <div>
  19. {
  20. list.map(item=><div key={ item.id } >
  21. { /* 做一些渲染页面的操作.... */ }
  22. </div>)
  23. }
  24. <button onClick={ ()=>this.getList() } >重新获取列表</button>
  25. </div>
  26. }
  27. }

这样页面请求数据,到数据更新,全部在当前组件发生,这个写法我不推荐,此时的数据走了一遍状态管理,最终还是回到了组件本身,显得很鸡肋,并没有发挥什么作用。在性能优化上到不如直接在组件内部请求数据。

不会合理使用状态管理

还有的同学可能这么写。

  1. class Text extends React.Component{
  2. constructor(prop){
  3. super(prop)
  4. this.state={
  5. list:[],
  6. }
  7. }
  8. async componentDidMount(){
  9. const { data , code } = await getList()
  10. if(code === 200){
  11. /* 获取的数据有可能是不常变的,多个页面需要的数据 */
  12. this.setState({
  13. list:data
  14. })
  15. }
  16. }
  17. render(){
  18. const { list } = this.state
  19. return <div>
  20. { /* 下拉框 */ }
  21. <select>
  22. {
  23. list.map(item=><option key={ item.id } >{ item.name }</option>)
  24. }
  25. </select>
  26. </div>
  27. }
  28. }

对于不变的数据,多个页面或组件需要的数据,为了避免重复请求,我们可以将数据放在状态管理里面。

如何使用状态管理

分析结构

我们要学会分析页面,那些数据是不变的,那些是随时变动的,用以下demo页面为例子:
优化 - 图18
如上 红色区域,是基本不变的数据,多个页面可能需要的数据,我们可以统一放在状态管理中,蓝色区域是随时更新的数据,直接请求接口就好。

总结

不变的数据,多个页面可能需要的数据,放在状态管理中,对于时常变化的数据,我们可以直接请求接口

八 海量数据优化-时间分片,虚拟列表

时间分片

时间分片的概念,就是一次性渲染大量数据,初始化的时候会出现卡顿等现象。我们必须要明白的一个道理,js执行永远要比dom渲染快的多。 ,所以对于大量的数据,一次性渲染,容易造成卡顿,卡死的情况。我们先来看一下例子

  1. class Index extends React.Component<any,any>{
  2. state={
  3. list: []
  4. }
  5. handerClick=()=>{
  6. let starTime = new Date().getTime()
  7. this.setState({
  8. list: new Array(40000).fill(0)
  9. },()=>{
  10. const end = new Date().getTime()
  11. console.log( (end - starTime ) / 1000 + '秒')
  12. })
  13. }
  14. render(){
  15. const { list } = this.state
  16. console.log(list)
  17. return <div>
  18. <button onClick={ this.handerClick } >点击</button>
  19. {
  20. list.map((item,index)=><li className="list" key={index} >
  21. { item + '' + index } Item
  22. </li>)
  23. }
  24. </div>
  25. }
  26. }

我们模拟一次性渲染 40000 个数据的列表,看一下需要多长时间。
image.gif
我们看到 40000 个 简单列表渲染了,将近5秒的时间。为了解决一次性加载大量数据的问题。我们引出了时间分片的概念,就是用setTimeout把任务分割,分成若干次来渲染。一共40000个数据,我们可以每次渲染100个, 分次400渲染。

  1. class Index extends React.Component<any,any>{
  2. state={
  3. list: []
  4. }
  5. handerClick=()=>{
  6. this.sliceTime(new Array(40000).fill(0), 0)
  7. }
  8. sliceTime=(list,times)=>{
  9. if(times === 400) return
  10. setTimeout(() => {
  11. const newList = list.slice( times , (times + 1) * 100 ) /* 每次截取 100 个 */
  12. this.setState({
  13. list: this.state.list.concat(newList)
  14. })
  15. this.sliceTime( list ,times + 1 )
  16. }, 0)
  17. }
  18. render(){
  19. const { list } = this.state
  20. return <div>
  21. <button onClick={ this.handerClick } >点击</button>
  22. {
  23. list.map((item,index)=><li className="list" key={index} >
  24. { item + '' + index } Item
  25. </li>)
  26. }
  27. </div>
  28. }
  29. }

效果
image.gif
setTimeout 可以用 window.requestAnimationFrame() 代替,会有更好的渲染效果。我们demo使用列表做的,实际对于列表来说,最佳方案是虚拟列表,而时间分片,更适合热力图,地图点位比较多的情况。

虚拟列表

笔者在最近在做小程序商城项目,有长列表的情况, 可是肯定说 虚拟列表 是解决长列表渲染的最佳方案。无论是小程序,或者是h5 ,随着 dom元素越来越多,页面会越来越卡顿,这种情况在小程序更加明显 。稍后,笔者讲专门写一篇小程序长列表渲染缓存方案的文章,感兴趣的同学可以关注一下笔者。
虚拟列表是按需显示的一种技术,可以根据用户的滚动,不必渲染所有列表项,而只是渲染可视区域内的一部分列表元素的技术。正常的虚拟列表分为 渲染区,缓冲区 ,虚拟列表区。
如下图所示。
优化 - 图21
为了防止大量dom存在影响性能,我们只对,渲染区和缓冲区的数据做渲染,,虚拟列表区 没有真实的dom存在。缓冲区的作用就是防止快速下滑或者上滑过程中,会有空白的现象。

react-tiny-virtual-list

react-tiny-virtual-list 是一个较为轻量的实现虚拟列表的组件。这是官方文档。

  1. import React from 'react';
  2. import {render} from 'react-dom';
  3. import VirtualList from 'react-tiny-virtual-list';
  4. const data = ['A', 'B', 'C', 'D', 'E', 'F', ...];
  5. render(
  6. <VirtualList
  7. width='100%'
  8. height={600}
  9. itemCount={data.length}
  10. itemSize={50} // Also supports variable heights (array or function getter)
  11. renderItem={({index, style}) =>
  12. <div key={index} style={style}> // The style property contains the item's absolute position
  13. Letter: {data[index]}, Row: #{index}
  14. </div>
  15. }
  16. />,
  17. document.getElementById('root')
  18. );

手写一个react虚拟列表

  1. let num = 0
  2. class Index extends React.Component<any, any>{
  3. state = {
  4. list: new Array(9999).fill(0).map(() =>{
  5. num++
  6. return num
  7. }),
  8. scorllBoxHeight: 500, /* 容器高度(初始化高度) */
  9. renderList: [], /* 渲染列表 */
  10. itemHeight: 60, /* 每一个列表高度 */
  11. bufferCount: 8, /* 缓冲个数 上下四个 */
  12. renderCount: 0, /* 渲染数量 */
  13. start: 0, /* 起始索引 */
  14. end: 0 /* 终止索引 */
  15. }
  16. listBox: any = null
  17. scrollBox : any = null
  18. scrollContent:any = null
  19. componentDidMount() {
  20. const { itemHeight, bufferCount } = this.state
  21. /* 计算容器高度 */
  22. const scorllBoxHeight = this.listBox.offsetHeight
  23. const renderCount = Math.ceil(scorllBoxHeight / itemHeight) + bufferCount
  24. const end = renderCount + 1
  25. this.setState({
  26. scorllBoxHeight,
  27. end,
  28. renderCount,
  29. })
  30. }
  31. /* 处理滚动效果 */
  32. handerScroll=()=>{
  33. const { scrollTop } :any = this.scrollBox
  34. const { itemHeight , renderCount } = this.state
  35. const currentOffset = scrollTop - (scrollTop % itemHeight)
  36. /* translate3d 开启css cpu 加速 */
  37. this.scrollContent.style.transform = `translate3d(0, ${currentOffset}px, 0)`
  38. const start = Math.floor(scrollTop / itemHeight)
  39. const end = Math.floor(scrollTop / itemHeight + renderCount + 1)
  40. this.setState({
  41. start,
  42. end,
  43. })
  44. }
  45. /* 性能优化:只有在列表start 和 end 改变的时候在渲染列表 */
  46. shouldComponentUpdate(_nextProps, _nextState){
  47. const { start , end } = _nextState
  48. return start !== this.state.start || end !==this.state.end
  49. }
  50. /* 处理滚动效果 */
  51. render() {
  52. console.log(1111)
  53. const { list, scorllBoxHeight, itemHeight ,start ,end } = this.state
  54. const renderList = list.slice(start,end)
  55. return <div className="list_box"
  56. ref={(node) => this.listBox = node}
  57. >
  58. <div
  59. style={{ height: scorllBoxHeight, overflow: 'scroll', position: 'relative' }}
  60. ref={ (node)=> this.scrollBox = node }
  61. onScroll={ this.handerScroll }
  62. >
  63. { /* 占位作用 */}
  64. <div style={{ height: `${list.length * itemHeight}px`, position: 'absolute', left: 0, top: 0, right: 0 }} />
  65. { /* 显然区 */ }
  66. <div ref={(node) => this.scrollContent = node} style={{ position: 'relative', left: 0, top: 0, right: 0 }} >
  67. {
  68. renderList.map((item, index) => (
  69. <div className="list" key={index} >
  70. {item + '' } Item
  71. </div>
  72. ))
  73. }
  74. </div>
  75. </div>
  76. </div>
  77. }
  78. }

效果
优化 - 图22
具体思路
① 初始化计算容器的高度。截取初始化列表长度。这里我们需要div占位,撑起滚动条。
② 通过监听滚动容器的 onScroll事件,根据 scrollTop 来计算渲染区域向上偏移量, 我们要注意的是,当我们向下滑动的时候,为了渲染区域,能在可视区域内,可视区域要向上的滚动; 我们向上滑动的时候,可视区域要向下的滚动。
③ 通过重新计算的 endstart 来重新渲染列表。
性能优化点
① 对于移动视图区域,我们可以用 transform 来代替改变 top值。
② 虚拟列表实际情况,是有 start 或者 end 改变的时候,在重新渲染列表,所以我们可以用之前 shouldComponentUpdate 来调优,避免重复渲染。

开启HMR

在上一篇文章《React基础——快速搭建开发环境》中我们提到了如何迈上开发React的第一步,这一篇文章中我们就来谈谈如何提高开发效率开启Hot-Module-Replacement以及使用react-hot-loader

什么是HMR

Webpack官网上写到 Hot Module Replacement (HMR) exchanges, adds, or removes modules while an application is running, without a full reload.
Hot Module Replacement简单的来说就是当你修改了应用时候,不让整个应用刷新的东西。还没有概念?请看下面两幅图(图1实时更新,没有开启HMR;图2实时更新,开启HMR),仔细找找有什么区别?
优化 - 图23优化 - 图24
估计大多数同学没看出区别。。。 请注意浏览器上方的刷新按钮,图一当我们修改代码的时候整个页面刷新了,图二页面并没有刷新只是修改了对应的内容,这样无疑可以结余开发时间

如何开启HMR

其实开启HMR非常简单,我们只需要修改自己的程序入口,在我们的例子中是src/index.js文件,修改如下:

  1. import React from 'react';
  2. import ReactDOM from 'react-dom';
  3. import './index.css';
  4. import App from './App';
  5. import registerServiceWorker from './registerServiceWorker';
  6. ReactDOM.render(<App />, document.getElementById('root'));
  7. //加入下面的几行代码
  8. if (module.hot) {
  9. module.hot.accept('./App', () => {
  10. ReactDOM.render(<App />, document.getElementById('root'));
  11. })
  12. }
  13. registerServiceWorker();
  14. 复制代码

这里还需要提一下,如果使用了Redux(当然如果你是新手可以跳过关于Redux这段配置,短时间内还用不上),开启HMR需要在create store的时候进行修改:

  1. import { createStore } from 'redux'
  2. import rootReducer from './reducers'
  3. const configureStore = () => {
  4. const store = createStore(rootReducer)
  5. if (process.env.NODE_ENV !== "production") {
  6. if (module.hot) {
  7. module.hot.accept('./reducers', () => {
  8. store.replaceReducer(rootReducer)
  9. })
  10. }
  11. }
  12. return store
  13. }
  14. export default configureStore
  15. 复制代码

开启HMR之后,已经可以做到网页自动刷新了,但实际上还可以进一步提高开发效率,我们做个例子: 修改App.js如下:

  1. import React, { Component } from 'react';
  2. import logo from './logo.svg';
  3. import './App.css';
  4. class App extends Component {
  5. state = {
  6. count: 0
  7. }
  8. addHandler = () => {
  9. this.setState(prevState => {
  10. return { count: prevState.count+1 }
  11. })
  12. }
  13. render() {
  14. return (
  15. <div className="App">
  16. <header className="App-header">
  17. <img src={logo} className="App-logo" alt="logo" />
  18. <h1 className="App-title">Welcome to React</h1>
  19. </header>
  20. <p className="App-intro">
  21. change Count={this.state.count}
  22. <button onClick={this.addHandler}>Add</button>
  23. </p>
  24. </div>
  25. );
  26. }
  27. }
  28. export default App;
  29. 复制代码

之后运行代码,点击Add,此时count已经变成1了,我们再修改文字,可以看到count的数值又变成0了,如果我们可以让count的数值不受影响该多好呀?下面我们就来使用react-hot-loader来实现这一点

使用react-hot-loader

react-hot-loader是在webpack HMR基础上增加了刷新页面时候依然应用保留状态的一个插件
首先我们需要调用yarn eject来弹出配置webpack的配置信息,在新生成的config文件夹下找到webpack.config.dev.js找到下图所示代码片段,加入plugins: ["react-hot-loader/babel"] 的配置(这里实际上是给webpack增加了一个plugin插件的配置)

  1. // Process JS with Babel.
  2. {
  3. test: /\.(js|jsx|mjs)$/,
  4. include: paths.appSrc,
  5. loader: require.resolve('babel-loader'),
  6. options: {
  7. // This is a feature of `babel-loader` for webpack (not Babel itself).
  8. // It enables caching results in ./node_modules/.cache/babel-loader/
  9. // directory for faster rebuilds.
  10. cacheDirectory: true,
  11. //加入下面这一句
  12. plugins: ["react-hot-loader/babel"]
  13. },
  14. },
  15. 复制代码

其次修改我们的根组件App.js如下:

  1. import { hot } from 'react-hot-loader';
  2. //这里是组件的定义,用省略号表示
  3. ...
  4. export default hot(module)(App);
  5. 复制代码

这里说明一下这里的hot实际上实现了上述我们谈到的HMR的功能以及保存状态的功能,因此我们的index.js文件下不用重复设置HMR的代码了
刷新页面,点击Add,此时count已经变成1了,我们再修改文字,可以看到count的数值并没有受到影响
优化 - 图25
这样在实际的项目中,可以保存状态又可以帮我们省去很多开发时间
更多学习链接:
hot-module-replacement
Github CRA中开启HMR的讨论
React-Hot-Loader Github主页