likes
comments
collection
share

一文讲明白useState极简版

作者站长头像
站长
· 阅读数 9

开篇

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

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

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

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

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

从一个问题开始讲解useState

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

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

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,下面给出核心代码,解释都会放到注释里面。

//...省略
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来做区分不用多说,不知道的还是回到我开题说的再去看看整个渲染流程。

ReactCurrentDispatcher.current =
    current === null || current.memoizedState === null
    ? HooksDispatcherOnMount
    : HooksDispatcherOnUpdate;

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

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

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

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

拓展

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