React 中监听并存储表单内存问题
在 React 中,可变的数据往往存放在 state 中
这种存储方式可以应对大部分情况,但是在某些特殊情况下会导致问题
起因
在编辑博客时,浏览器不断报错,提示out of memory
,搜了下并没有发现有类似的讨论
使用调试工具看了下内存占用,发现随便打几个字轻轻松松就超过 3G。
虽然 Chrome 以耗费内存著称,但是没道理打几个字就占用 3 个G,所以针对整个问题,进行了各种测试。
确定问题
首先,这个问题在之前的博客上是不存在的,基本上可以确定问题出在 Monaco Editor 上
所以单独把 Monaco Editor 拉出来建项目测试,发现输入并没有什么问题。而且使用的框架 Demo 测试也没有问题
而本地跑博客的后端依旧爆炸,那么怀疑和博客本身的某个东西有问题。
自然而言要怀疑的对象是是 Ant Design 的 Form
,但是去看了下代码,并没有发现有什么问题
那么要找到问题的原因,就要先稳定重现
在 CodeSandbox 上经过各种测试,可以发现如下问题:
- 编辑框内容很长时才会出现内存飙升(体现为打一个字多几十兆)
- 当内存飙升时,打字开始卡顿(但是电脑本身并未卡顿)
- 打中文比打英文更容易出问题
- 复制粘贴大文本没问题,只有打字会有问题
那么就可以稳定重现了:生成 2w 个中文,然后打中文进去
接着,开始删东西,确定是哪些组件的问题。当删去Form.Item
时,可以看到内存不再飙升,那么真的是Form
的问题么?
要控制变量确定这一问题,就要实现一个类似功能的东西。虽然 Ant Design 的 From
很复杂,但是涉及和编辑框交互的只有监听内容的部分。那么就监听onChange
存下来,看看有没有问题
测试发现,仅仅是无脑onChange={value=>this.setState({value})}
,就可以稳定触发
那么就可以确定就是监听这里有问题了
再进一步测试使用防抖函数,确保每一秒只存入 state 一次
结果与预期一样,毫无问题
原因猜想
首先大文本存储肯定是需要时间的,由于存储时不是增量存储,所以相当于每次都要全量写入所有文本
而中文比英文触发快的原因是:在打字时,如果要打中文,输入框内会先出现对应的拼音(以及分隔符),再算上最后确认中文后又快速把拼音编程中文,一秒钟完全可以触发十几次onChange
,相对于英文触发的更多
看起来一切都理所当然,本身就不应该监听并存储大数据。
但是测试过程中并不是那么直觉,比如最开始提到的:老版本的 TextArea
并没有这种问题出现,因此更容易将问题定向到 Monaco Editor 本身,而非使用的方式。
(而且一篇博客最多也就几万字,那么多编辑器都没事)
解决方案
解决方案其实前面也已经提到过了,要监听变化就加一个防抖,给浏览器足够的处理时间
要获取内容只在最后获取,而非中间保持数据同步