本文首发于公众号:符合预期的CoyPan
前两篇文章,深刻了Hook的源码。本文将以useState
和useEffect
为主,总结一下Hook的相关内容,javascript
先贴出示例代码:java
const App = () => { const [title, setTitle] = useState(''); const [fontSize, setFontSize] = useState('13'); useEffect(() => { document.title = title; },[title]); useEffect(() => { document.body.style.fontSize = `${fontSize}px`; }, [fontSize]); return <React.Fragment> <input value={title} onChange={e => setTitle(e.target.value)} /> <input value={fontSize} onChange={e => setFontSize(e.target.value)} /> <p>Hello World</p> </React.Fragment> };
组件首次挂载的时候,依次执行四行useXXX
的代码,生成4个hook对象,按照顺序造成完整的hook链,挂载在Fiber对象的memorizedState
属性上。hook对象上的memorizedState
属性表示当前hook对象的值。对于effect-hook
,其memorizedState
属性的值为一个对象(effect对象),包含了一个指向下一下effect-hook
的effect对象的指针,会造成一个环形链表。react
对于state-hook
,其setState
函数经过闭包保存了对应的hook对象的引用。调用setState
对状态值进行更新时,每调用一次setState
,都会修改hook
对象上的queue
属性。queue
属性保存着hook的更新信息。setState
执行完后,对应的hook对象上的memorized
值就是最新的了。git
对于effect-hook
,每次组件渲染时,都会进入effect-hook
的逻辑:首先判断deps
数组内的值较上一次有无变化。若是无变化,则会给该effect打上忽略的tag。若是跟上一次组件渲染时有变化,最终会先执行destory
函数(即useEffect
中return的函数),而后执行create
函数。github
对于反作用,react每次都会生成新的effect链,若是deps
数组内的值变化了,那么新的effect对象还会赋值给effect-hook
的memorizedState
属性。数组
本例中,假设咱们在title那里,输入了hello,组件更新完成后,Fiber节点上的hook信息以下:闭包
稍微总结一下,关于react hook,咱们须要知道如下几点:函数
一、react hook大量采用了链表的结构。spa
二、代码中hook的书写顺序,决定了最终hook在fiber上的存储顺序。指针
三、组件每次渲染的时候,hook的代码都会执行一次,而且将上一次的hook对象都clone一遍,返回最新的hook的state以及对应的更新函数。在state的更新函数中,以闭包的形式保存了对应hook对象的引用。
至此,react hook源码解析系列暂时告一段落了。尽管有许多地方没有我还没彻底理解,有的地方也没有讲述清楚,可是对react hook的总体实现方式已经有了进一步深刻的理解了。符合预期。最后,推荐一篇hook相关的文章:
https://github.com/shanggqm/b...