切换主题
深入理解useState
基础用法
可以将useState
官方文档中基础用法总结如下:
javascript
const [cnt, setCnt] = useState(0)
const handleClick = () => {
// 自动批处理:减少函数组件的调用次数
setCnt(cnt + 1)
setCnt(cnt + 1)
// 取消自动批处理 -- flushSync:需要谨慎使用
flushSync(() => setCnt(cnt + 1))
flushSync(() => setCnt(cnt + 1))
// 取消自动批处理 -- 异步代码
setTimeout(() => {
setCnt(cnt + 1)
setCnt(cnt + 1)
}, 0)
// 取消自动批处理 -- 回调函数:也可用于获取更新后的值
setCnt((cnt) => cnt + 1)
setCnt((cnt) => cnt + 1)
}
// 惰性初始化:lazyComputation函数只会在初始渲染时被调用
const lazyComputation = () => {
const initialState = someComputation(props)
return initialState
}
const [state, setState] = useState(lazyComputation)
注意事项
不能在循环、条件或嵌套函数中调用
Hook
:Hook
的调用顺序在每次渲染中必须保持一致,否则会导致状态混乱。状态更新是异步的:
React
可能会批量更新状态,因此在更新状态后立即读取状态可能得到的是旧值。
获取新值
由于useState
是异步的,所以可以通过下面四种方法获取更新后的值:
- 在组件重新渲染后获取新值:
jsx
import React, { useState } from 'react'
const Counter = () => {
const [cnt, setCnt] = useState(0)
const handleClick = () => {
setCnt(cnt + 1) // 更新状态
}
return (
<div>
<p>当前 cnt: {cnt}</p> {/* 这里会显示更新后的值 */}
<button onClick={handleClick}>增加</button>
</div>
)
}
- 使用
useEffect
监听状态变化:
jsx
import React, { useState, useEffect } from 'react'
const Counter = () => {
const [cnt, setCnt] = useState(0)
useEffect(() => {
console.log('cnt 已更新:', cnt) // 每次 cnt 更新后都会触发
}, [cnt]) // 监听 cnt 的变化
const handleClick = () => {
setCnt(cnt + 1)
}
return (
<div>
<p>当前 cnt: {cnt}</p>
<button onClick={handleClick}>增加</button>
</div>
)
}
- 函数式更新中获取新值:
jsx
import React, { useState } from 'react'
const Counter = () => {
const [cnt, setCnt] = useState(0)
const handleClick = () => {
setCnt((prevCnt) => {
const newCnt = prevCnt + 1
console.log('新 cnt:', newCnt) // 这里可以获取到新值
return newCnt
})
}
return (
<div>
<p>当前 cnt: {cnt}</p>
<button onClick={handleClick}>增加</button>
</div>
)
}
- 使用
useRef
存储最新值:
jsx
import React, { useState, useRef, useEffect } from 'react'
const Counter = () => {
const [cnt, setCnt] = useState(0)
const countRef = useRef(cnt)
useEffect(() => {
countRef.current = cnt // 每次 cnt 更新后同步到 ref
}, [cnt])
const handleClick = () => {
setCnt(cnt + 1)
console.log('最新 cnt:', countRef.current) // 这里可以获取到最新值
}
return (
<div>
<p>当前 cnt: {cnt}</p>
<button onClick={handleClick}>增加</button>
</div>
)
}
实现原理
以下是一个简单实现,只是帮助理解其工作原理:
javascript
let state // 当前状态
let setters // 更新函数数组
let stateIdx = 0 // 当前状态索引
function useState(initState) {
const curIdx = stateIdx
// 初始化
if(state === undefined) {
state = []
}
// 初始化赋值
if(state[curIdx] === undefined) {
state[curIdx] = typeof initState === 'function'
? initState()
: initState
}
// 初始化setter函数
const setState = (newState) => {
state[curIdx] = typeof newState === 'function'
? newState(state[curIdx])
: newState
render() // 触发重新渲染
}
setter[curIdx] = setState
stateIdx++
return [state[curIdx], setState]
}
function render() {
stateIdx = 0 // 重置索引
// 重新渲染组件
}
事务机制
React
引入事务机制即批量处理机制(默认异步)的原因?
为了优化性能,减少不必要的渲染。
批量处理(batch
)主流程:
null
事务机制的原理如下:
执行定义的方法
updateMsg
时,先声明isBatchingUpdates
,并设置为true
;updateMsg
函数内部按序执行到setMsg
时,此时的isBatchingUpdate
初始值为true
,命中batchUpdate
,走主流程的中Y
分支:
jsx
const Component = () => {
const [msg, setMsg] = useState('init')
const updateMsg = () => {
// 1.开始时
// isBatchingUpdate = true,命中batchUpdate
setMsg('update')
// 2.结束时
// isBatchingUpdate = false
}
retrun({
// JSX
})
}
- 若
setMsg
作为异步函数的回调函数会等待同步代码执行完
(同EventLoop
),再执行,而此时的isBatchingUpdate
为false
,未命中batchUpdate
,走主流程的中N
分支:
jsx
const Component = () => {
const [msg, setMsg] = useState('init')
const updateMsg = () => {
// 1.开始时
// isBatchingUpdate = true
setTimeout(() => {
// 此时isBatchingUpdate = false,未命中batchUpdate
setMsg('update')
}, 0)
// 2.结束时
// isBatchingUpdate = false
/* 等价于在此时执行setMsg */
// setMsg('update')
}
retrun({
// JSX
})
}
总结
同步更新:在
React
生命周期方法、React
事件处理函数中调用setState
或useState
时,更新是异步的,因为React
会将这些更新放入队列中批量处理。异步更新:在
setTimeout
、setInterval
或自定义DOM
事件处理函数中调用setState
或useState
时,更新是同步的,因为此时React
不会进行批量更新。