reacthooks用法(详解React Hooks是如何工作的)
reacthooks用法
详解React Hooks是如何工作的目录
- 1. React Hooks VS 纯函数
- 2. 简单 myUseState
- 3. 改进 myUseState
- 4. 实现原理引发的 Hooks 规则
React Hook 说白了就是 React V18.6 新增的一些 API,API的本质就是提供某种功能的函数接口。因此,React Hooks 就是一些函数,但是 React Hooks 不是纯函数。
什么是纯函数呢?就是此函数在相同的输入值时,需产生相同的输出,并且此函数不能影响到外面的数据。
简单理解就是函数里面不能用到在外面定义的变量,因为如果用到了外面定义的变量,当外面的变量改变时会影响函数内部的计算,函数也会影响到外面的变量。
对于 React Hooks 提供的函数 API,恰恰就不是纯函数。
来看一个 useState 的使用语句 const [count, setCount] = useState(0),使用 useState 函数得到的结果并不是全都一样的,因为如果 useState(0) 每次得到的结果都是一样的,那 count 值就永远不会改变了,那 count 所在的页面就永远不会改变,和我们看到的结果就不一样了。由此可知,React Hooks 都不是纯函数,也就是说 Hooks 用到了函数外的变量。
那么是什么特性让 React Hooks 一定不能是纯函数呢?实际上是 React 框架和函数组件本身决定的。我们知道,React 页面渲染的原理就是通过每次 render 得到新的虚拟 DOM ,然后进行 DOM Diff 来渲染页面。而 React 的函数组件是通过执行整个函数得到一个虚拟 DOM。因此在每次页面渲染 render 时,在函数组件内部的所有语句都会重新执行一次。如果在函数组件内部使用的 React Hooks 是纯函数的话,就不会在每次渲染后得到不同的虚拟 DOM 了。
React 规定: 所有 React 组件都必须是纯函数,并禁止修改其自身 props 。
因此在 React V16.8 之前 React Hooks 还没出来的时候,函数组件因为是纯函数,只能返回一个固定的虚拟 DOM,不能包含状态,也不支持生命周期方法。因此,当时仅仅是支持函数组件,但函数组件相比于类组件限制太多,函数组件无法取代类组件,也没类组件好用。
React 希望组件是简单的而不是复杂的,React 认为组件的最佳写法应该是函数,而不是类。因此 React 就新增了 React Hooks,Hook 就是钩子的意思,是 React 提供给函数组件在需要外部功能和数据状态时将其 “钩” 进去,从而完善函数组件,使其能完全代替类组件。
React 的函数组件只能是纯函数,那么每次事件发生时重新 render 函数组件时得到不同的虚拟 DOM 的事就完全交给了 React Hooks,那么 React Hooks 是如何做到的呢?下面就手动实现一个 useState,useState 的具体细节肯定不是这样的,但原理和思路是一样的。
2. 简单 myUseStateReact.useState 的第一次执行是将初始值赋予给一个 _state,之后的每次重新 render 时就是读取 _state 的值。[state, setState] 中的 setState 做的事就是改变 _state 的值,然后重新渲染页面。
根据这个原理实现 myUseState 函数如下:
import React from 'react'; import ReactDOM from 'react-dom'; let _state function myUseState(initialValue){ if(_state === undefined){ _state = initialValue } const setState = (newValue)=>{ _state = newValue render() } return [_state, setState] } function render(){ ReactDOM.render(<App/>,document.getElementById('root')); } function App(){ const [n, setN] = myUseState(0) return ( <li> n: {n} <button onClick={() => setN(n+1)}>+1</button> </li> ) } ReactDOM.render(<App/>,document.getElementById('root'));
上述实现的 myUseState 存在 bug,当在函数组件内用到两次 myUseState 时就会出现问题了,二者共用一个 _state 会出现混乱。
因此需要将上述实现进行改进,改进的思路就是将 _state 定义为一个数据或者是对象,由于我们在函数使用时只传了一个数值,无法确定键值,因此只能使用数据。改进如下:
import React from 'react'; import ReactDOM from 'react-dom'; let _state = [] let index = 0 function myUseState(initialValue){ const currentIndex = index if(_state[currentIndex] === undefined){ _state[currentIndex] = initialValue } const setState = (newValue)=>{ _state[currentIndex] = newValue render() } index++ return [_state[currentIndex], setState] } function render(){ index = 0 ReactDOM.render(<App/>,document.getElementById('root')); } function App(){ const [n, setN] = myUseState(0) const [m, setM] = myUseState(0) return ( <li> n: {n} <button onClick={() => setN(n+1)}>+1</button> <br/> m: {m} <button onClick={() => setM(m+1)}>+1</button> </li> ) } ReactDOM.render(<App/>,document.getElementById('root'));
上述实现的 myUseState 肯定不是 React.useState 的具体实现代码,但实现原理是一致的。myUseState 函数封装了函数组件内的数据状态,并对该状态进行管理,以暴露出相关的操作接口的方式提供给函数组件使用。
这样一来,函数组件就和其数据状态分离了,函数组件只负责返回虚拟 DOM 本身就可以了,对于数据状态的管理完全交给其 “钩” 住的 React.useState Hook 就可以了。
从上述的实现思路可以发现,React Hooks 的实现其实是基于 全局变量 和 闭包 原理实现的特殊函数。
但是,正是因为这样的实现方式,限制了 React Hooks 的使用必须是 只在顶层调用Hook,意思就是说 不要在循环,条件或嵌套函数中调用 Hook,如果在 if 条件句中使用了 Hook, 导致组件每次渲染生成时 React.useState 语句的执行次数不对,就会打乱 index 的计数,从而导致数据维护的错误。
上述的实现原理依赖于 index 的正确计数,因此 React 依赖于调用 Hooks 的顺序,
以上就是详解React Hooks是如何工作的的详细内容,更多关于详解React Hooks的资料请关注开心学习网其它相关文章!
- react重点和难点(关于React状态管理的三个规则总结)
- react常用设计模式(提高React界面性能的十个技巧)
- vue3 props用法(vue3组合API中setup、 ref、reactive的使用大全)
- react定时任务(手把手带你用React撸一个日程组件)
- react源码教程(详解React 代码共享最佳实践方式)
- react高阶组件怎么用(React 高阶组件HOC用法归纳)
- vue react和angular(详解React Angular Vue三大前端技术)
- react性能优化是哪个周期(React 并发功能体验前端的并发模式)
- react基础知识入门(浅谈React 的引入)
- react native常用组件(react native环境安装流程)
- react怎么绑定state(react纯函数组件setState更新页面不刷新的解决)
- react常见问题(React编程中需要注意的两个错误)
- reactnative示例代码(React Native项目框架搭建的一些心得体会)
- vscode react插件(基于visual studio code + react 开发环境搭建过程)
- reacthooks基础使用(React 小技巧教你如何摆脱hooks依赖烦恼)
- reacthooks用法(详解React Hooks是如何工作的)
- 演员陈创,火于 哮天犬 ,颠峰于 福贵 ,现状却令人唏嘘(演员陈创火于哮天犬)
- 幼小衔接-20以内看图读数 写数 数的组成练习题(幼小衔接-20以内看图读数)
- 你只要花上20天记单词,英语成绩就能从57提到100(你只要花上20天记单词)
- 夕云天际飞,亢龙化太极(夕云天际飞亢龙化太极)
- 爱情可以当饭吃吗(怎么回复)
- 高考数学题(高考数学题基础题占多少分)
热门推荐
- 有哪些优秀的web服务器(几款比较好的WEB服务器软件推荐)
- css3浮动教程(CSS3使用多列制作瀑布流)
- DateTime.ToString 格式化时间字符串
- docker怎么设置redis(docker安装redis并以配置文件方式启动详解)
- 优化网站的速度的方法
- kubernetes最优pod数(详解kubernetes pod的编排和生命周期)
- python矩阵怎么生成(python实现矩阵打印)
- docker进入容器的方法(docker容器里面的root权限获取方法)
- ASP.NET中application对象的用法
- pythonsocket建立多用户通讯(Python socket实现多对多全双工通信的方法)
排行榜
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9