React 架构的演变 – Hooks 的实践

React Hooks 可以说完全颠覆了之前 Class Component 的写法,进一步增强了状态复用的能力,让 Function Component 也具有了内部状态,对于我个人来说,更加喜欢 Hooks 的写法。当然如果你是一个使用 Class Component 的老手,初期上手时会觉得很苦恼,毕竟之前沉淀的很多 HOC、Render Props 组件基本没法用。而且之前的 Function Component 是无副作用的无状态组件,现在又能通过 Hooks 引入状态,看起来真的很让人疑惑。Function Component 的另一个优势就是可以完全告别 this ,在 Class Component 里面 this 真的是一个让人讨厌的东西?? 。

Hook 如何与组件关联

在之前的文章中多次提到,Fiber 架构下的 updateQueue、effectList 都是链表的数据结构,然后挂载的 Fiber 节点上。而一个函数组件内所有的 Hooks 也是通过链表的形式存储的,最后挂载到 fiber.memoizedState 上。


  1. function App() { 
  2.   const [num, updateNum] = useState(0) 
  3.  
  4.   return <div 
  5.     onClick={() => updateNum(num => num + 1)} 
  6.   >{ num }</div> 
  7.  
  8. export default App 

我们先简单看下,调用 useState 时,构造链表的过程:


  1. var workInProgressHook = null 
  2. var HooksDispatcherOnMount = { 
  3.   useState: function (initialState) { 
  4.     return mountState(initialState) 
  5.   } 
  6.  
  7. function function mountState(initialState) { 
  8.   // 新的 Hook 节点 
  9.   var hook = mountWorkInProgressHook() 
  10.   // 缓存初始值 
  11.   hook.memoizedState = initialState 
  12.   // 构造更新队列,类似于 fiber.updateQueue 
  13.   var queue = hook.queue = { 
  14.     pending: null
  15.     dispatch: null
  16.     lastRenderedState: initialState 
  17.   } 
  18.   // 用于派发更新 
  19.   var dispatch = queue.dispatch = dispatchAction.bind( 
  20.     null, workInProgress, queue 
  21.   ) 
  22.   // [num, updateNum] = useState(0) 
  23.   return [hook.memoizedState, dispatch] 
  24.  
  25. function mountWorkInProgressHook() { 
  26.   var hook = { 
  27.     memoizedState: null
  28.     baseState: null
  29.     baseQueue: null
  30.     queue: null
  31.     nextnull 
  32.   } 
  33.  
  34.   if (workInProgressHook === null) { 
  35.     // 构造链表头节点 
  36.     workInProgress.memoizedState = workInProgressHook = hook 
  37.   } else { 
  38.     // 如果链表已经存在,在挂载到 next 
  39.     workInProgressHook = workInProgressHook.next = hook 
  40.   } 
  41.  
  42.   return workInProgressHook 
【声明】:芜湖站长网内容转载自互联网,其相关言论仅代表作者个人观点绝非权威,不代表本站立场。如您发现内容存在版权问题,请提交相关链接至邮箱:bqsm@foxmail.com,我们将及时予以处理。

相关文章