时间:2021-05-02
最近工作中遇到个有意思的问题,记录下从问题发现到解决的过程。
这个问题涉及知识点包括:
某个需求需要引入一个第三方组件库。
当引入组件库中的函数组件A后,React运行时报错:
从React文档了解到,这是由于「错误使用Hooks造成的」。
官网给出的可能的错误原因有3种:
1.React和ReactDOM版本不匹配
需要v16.8以上版本的ReactDOM才支持Hooks。
我们项目使用的是v17.0.2,不属于这个原因。
2.打破了Hooks的规则
Hooks只能在函数组件或自定义Hooks顶层调用。
翻看A组件源码,报错的是一个顶层调用的useRef:
不属于这个原因。
3.重复的React
载录自React文档:
读起来好绕,看起来这条的嫌疑最大。
在报错的useRef中打上断点,发现其来自于:
http://localhost:8081/Users/项目目录/node_modules/组件库/node_modules/react/cjs/react.development.js
在项目里其他调用Hooks但是未报错的地方打上断点,发现资源来自于:
http://localhost:8081/Users/项目目录/node_modules/react/cjs/react.development.js
报错的useRef和项目其他Hooks引用了不同的react.development.js。
翻看「组件库」的package.json,发现他将react与react-dom作为dependencies安装:
这样会在「组件库」目录的node_modules下创建这两个依赖。
作为一个「组件库」,这么做显然是不合适的。
最好的做法是将这两个依赖作为peerDependencies,即将其作为外部依赖。
这样,当我们引入「组件库」时,「组件库」会使用我们项目中的react与react-dom,而不是自己安装一份。
但是我没有这个「组件库」的权限,只能在自己项目中做文章。
在package.json文档中提供了一个配置项:resolutions,可以临时解决这个问题。
resolutions允许你复写一个在项目node_modules中被嵌套引用的包的版本。
在我们项目的package.json中作出如下修改:
这样,项目中用到的这两个依赖都会使用resolutions中指定的版本。
不管是「组件库」还是我们的项目代码中的react与react-dom,都会指向同一个文件。
现在问题是临时解决了,但是造成问题的原因是什么?
让我们深入Hooks源码内部来寻找答案。
首先让我们思考2个问题:
当我们在一个Hooks内部调用其他Hooks时会报开篇提到的错误。
比如如下代码就会报错:
Hooks只是函数,他如何感知到自己在另一个Hooks内部执行?
就如上例子,useRef如何感知到自己在useEffect的回调函数中执行?
再看另一个问题,我们知道classComponent有componentDidMount与componentDidUpdate两个生命周期函数区分mount时与update时。
那么Hooks作为函数,怎么区分当前是mount时还是update时?
显然,Hooks源码内部存在一种机制,能够感知当前执行的上下文环境。
在浏览器环境,我们会引用react与reactDOM两个包。
其中,在react包的代码中存在一个变量ReactCurrentDispatcher。
他的current参数指向当前正在使用的Hooks上下文:
同时,在reactDOM中,在程序运行过程中,ReactCurrentDispatcher.current会根据当前上下文环境指向不同引用。
比如:
当处在DEV环境mount时,ReactCurrentDispatcher.current会指向HooksDispatcherOnMountInDEV。
当处在DEV环境update时,ReactCurrentDispatcher.current会指向HooksDispatcherOnUpdateInDEV。
再来看useRef的定义:
内部调用的是dispatcher.useRef。
dispatcher即ReactCurrentDispatcher.current。
这就是Hooks能区分mount与update的原因。
同理,DEV环境,当一个Hooks在执行时,ReactCurrentDispatcher.current会指向引用 —— InvalidNestedHooksDispatcherOnUpdateInDEV。
在这种情况下再调用的Hooks,比如如下useRef:
内部都会执行warnInvalidHookAccess报错,提示自己在别的Hooks内执行了。
到这里我们终于知道开篇提到的问题发生的本质原因:
总结
通过分析这个问题,加深了对package.json以及Hooks源码的理解。
不知道Hooks感知上下文的实现思路对你有没有启发呢?
原文地址:https://mp.weixin.qq.com/s/AJCtEXKDp-UKM7Lq4scxSQ
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
前言:在使用pycharm学习python的时候,经常需要第三方库,没有第三方库程序就会报错,pycharm也会提醒你要安装所需要的库,安装第三方库的时候往往就
如果要用ASP来作一个FTP文件上传的页面,我想很多人立刻就会想到要用第三方开发的组件,利用第三方的组件,虽然开发起来相对比较容易。但一般来说,免费下载的第三方
在工作中我们常常要用到tab组件,如果有用第三方组件库的话一般都会有这个组件,但如果没有使用第三方组件库,或者想要自定义tab,那么或许这个无依赖的tab组件将
mybatis本身没有提供日志的实现,引入的是第三方组件。mybatis支持多个第三方日志插件,优先级由低到高为slf4J、commonsLoging、Log4
b选择合适的第三方物流企业,是企业与第三方物流企业进行战略合作的前提。选择第三方物流企业时需要考虑的因素很多,主要有以下三方面:1.成本费用在决定选择第三方物流