一文讲明白useState极简版

开篇

本文要求读者最起码大致了解:

  • 基本fiber更新流程
  • 基本fiber树渲染

不需要了解的很深刻也不需要了解完所有的更新渲染情况,但基本的流程需要已经了解过。如果没学习过的同学推荐先去学习一下上面两点,上面两点是整个react的骨架,useState等hook只是基于整个骨架运行的。

然后需要说一下的本文的react版本是17,react18将并发以及可中断渲染正式投入使用,所以react18的useState比react17复杂一些。有些人可能会问不是react17就有可中断渲染和并发了么,事实上在react17中可中断渲染虽然实现, 但是并没有在稳定版暴露出 api。

综上本文基于react17来进行讲解。什么?为啥不讲18,因为怕大家听不懂所以从相对简单的17开始讲,绝对不是作者自己不懂哈。

从一个问题开始讲解useState

直接从头开始讲整个useState我认为稍微有点枯燥并且东西太多一时半会儿get不到最核心的点。本文以下面这个问题入手开始讲useState我认为让读者有一个核心主线会更好理解,下面给出问题:

有以下代码,问一般情况下haha是何时执行

javascript 复制代码
const [data, setData] = useState(1);
cosnt haha = (now) => {return now + 1}
<div onClick={() => {setData(haha)}}>

有一种小白回答是,执行到setData的时候就开始执行,好吧,面试就这么答,答完就是,面试官:你还有什么问题要问我么。(狗头保命)

好吧,我们来认真回答这个问题,在17中,setState其实调用的是dispatchActiondispatchAction的参数从前往后三个分别是,当前useState所在的fiberhook.queue(存储update对象的环形链表)以及我们传入的haha函数。

入参说完了,来看看这个函数做了什么,我就不大片大片的贴源码了,给出链接大伙自己去康康,dispatchAction,其实就三件事情:

  • 创建update对象并将我们传入的haha函数作为他的action属性
  • update对象添加进hook.queue的环形链表中,最后在hook.queue.pending
  • 运行scheduleUpdateOnFiber发起新一轮调度

回到我们一开始的问题haha在何时执行,现在我们看看当前haha的位置是不是在:hook.queue.pending.action上,可以看到dispatchAction是不会执行haha的。好吧我们继续往下看,既然知道了haha的存放位置是一个环形链表,那看hook.queue什么时候被拿出来读取执行了。

由于执行了scheduleUpdateOnFiber所以我们又会走一遍fiber树构造和fiber树渲染的过程。我们知道这些要更新的状态肯定是在fiber树构造过程中更新的,所以视野聚焦到fiber树构造beginwork中renderWithHooks里的Component再次执行useState(1)的时候。

这里的useState实际上最后会调用updateReducer,这个函数会执行最终的haha,下面给出核心代码,解释都会放到注释里面。

javascript 复制代码
//...省略
function basicStateReducer<S>(state: S, action: BasicStateAction<S>): S {
  return typeof action === 'function' ? action(state) : action;
}
reducer = basicStateReducer;
if (update.eagerReducer === reducer) {
    newState = ((update.eagerState: any): S);
} else {
    const action = update.action;
    // action实际上就是我们的haha()
    newState = reducer(newState, action);
}
//...省略
hook.memoizedState = newState;
const dispatch: Dispatch<A> = (queue.dispatch: any);
return [hook.memoizedState, dispatch];

所以回到题目haha 是什么时候执行的,答:更新fiber树时执行updateReducer时执行的。

看到这里,就会有疑惑,那按照这个说法,我初次渲染的时候应该也执行了haha,但是为什么我的data在初次渲染结果是1呢?

两种useState

事实上,useState在初次渲染和二次更新的时候对应指向的函数是不同的,初次渲染时是mountState,二次更新时是updateState两者实现有所不同。updateState所做的事情就是我上文所说的那样,而mountState做的事情会有一些不同,先说一下react是怎么区分何时使用哪一种函数的,下面给出核心代码。为什么能靠currentmemoizedState来做区分不用多说,不知道的还是回到我开题说的再去看看整个渲染流程。

javascript 复制代码
ReactCurrentDispatcher.current =
    current === null || current.memoizedState === null
    ? HooksDispatcherOnMount
    : HooksDispatcherOnUpdate;

最后useState默认就会去ReactCurrentDispatcher.current上拿对应的函数。

好吧,现在再来说一下mountState做了什么,其实很简单mountState总的来说就是四件事:

  • 创建了一下hook对象
  • 初始化hook对象属性(queue、memoizedState等)
  • 设置basicStateReducer
  • 返回[当前状态, dispatch函数]

好吧到这里其实问题基本都解开了。

拓展

这里拓展一个问题,什么情况会使得update.eagerReducer === reducertrue,这是useState一个性能优化点,可以去研究研究哦。

相关推荐
web1350858863510 分钟前
前端node.js
前端·node.js·vim
m0_5127446412 分钟前
极客大挑战2024-web-wp(详细)
android·前端
潜意识起点35 分钟前
精通 CSS 阴影效果:从基础到高级应用
前端·css
奋斗吧程序媛40 分钟前
删除VSCode上 origin/分支名,但GitLab上实际上不存在的分支
前端·vscode
IT女孩儿1 小时前
JavaScript--WebAPI查缺补漏(二)
开发语言·前端·javascript·html·ecmascript
m0_748256563 小时前
如何解决前端发送数据到后端为空的问题
前端
请叫我飞哥@3 小时前
HTML5适配手机
前端·html·html5
@解忧杂货铺5 小时前
前端vue如何实现数字框中通过鼠标滚轮上下滚动增减数字
前端·javascript·vue.js
F-2H7 小时前
C语言:指针4(常量指针和指针常量及动态内存分配)
java·linux·c语言·开发语言·前端·c++
gqkmiss7 小时前
Chrome 浏览器插件获取网页 iframe 中的 window 对象
前端·chrome·iframe·postmessage·chrome 插件