Hooks 简化了 React 组件内部状态和副作用的管理。此外,可以将重复的逻辑提取到自定义 Hooks 中,以在整个应用程序中重复使用。Hooks 严重依赖于 JS 闭包。这就是为什么 Hooks 如此具有表现力和简单,但是闭包有时很棘手。
& Q$ ?% f( y |0 S 使用 Hooks 时可能遇到的一个问题就是过时的闭包,这可能很难解决。让我们从过时的装饰开始。然后,看看到过时的闭包如何影响 React Hooks,以及如何解决该问题。
5 m5 M6 e) J. B" K4 q7 A/ U5 S8 Q7 E- d 1.过时的闭包 1 w) D4 u/ e* Y* Y1 x9 f& {
工厂函数 createIncrement(incBy) 返回一个increment和log函数的元组。调用时,increment()函数将内部value增加incBy,而log()仅打印一条消息,其中包含有关当前value的信息:
9 x: o) E* X% F# q2 q& i; z h function createIncrement(incBy) {
let value = 0;
function increment() {
value += incBy;
console.log(value);
}
const message = `Current value is ${value}`; function log() { console.log(message); }
return [increment, log];
}
const [increment, log] = createIncrement(1);
increment(); // 1
increment(); // 2
increment(); // 3
// 不能正确工作!
log(); // "Current value is 0" [increment, log] = createIncrement(1)返回一个函数元组:一个函数增加内部值,另一个函数记录当前值。然后,increment()的3次调用将 value递增到3。最后,log()调用打印消息是 Current value is 0,这有点出乎意料的,因为此时 value 为 3 了。' _9 B( c* C3 |) O- R
log()是一个过时的闭包。闭包 log()捕获了值为 "Current value is 0"的 message 变量。即使 value 变量在调用increment()时被增加多次,message变量也不会更新,并且总是保持一个过时的值 "Current value is 0"。过时的闭包捕获具有过时值的变量。0 {8 o3 K6 t$ P( I! l; I
2.修复过时的闭包 6 c* }9 C9 t) [" }7 i w. R. x0 j
修复过时的log()问题需要关闭实际更改的变量:value的闭包。我们将语句 const message = ...; 移动到 log() 函数内部:
4 a7 I0 k3 `+ }! a( O function createIncrement(incBy) {
let value = 0;
function increment() {
value += incBy;
console.log(value);
}
function log() {
const message = `Current value is ${value}`; console.log(message);
}
return [increment, log];
}
const [increment, log] = createIncrement(1);
increment(); // 1
increment(); // 2
increment(); // 3
// Works!
log(); // "Current value is 3" 现在,在调用了 3 次 increment() 函数之后,调用 log() 记录了实际value:"Current value is 3"。2 r0 `- r7 M8 n) u" h1 X0 [
3. Hooks 中的过时闭包 4 c! g$ @, k, E* m7 W; d) {
3.1 useEffect() * }6 n: y9 k2 V5 I2 k
我们来看一下使用useEffect() 过时闭包的常见情况。在组件<WatchCount>中,useEffect() 中每2秒记录一次count的值:$ j' ~8 _; C6 S* u
function WatchCount() {
const [count, setCount] = useState(0);
useEffect(function() {
setInterval(function log() {
console.log(`Count is: ${count}`);
}, 2000);
}, []);
return (
<div> {count} <button onClick={() => setCount(count + 1) }> Increase </button> </div>
);
} 打开事例(https://codesandbox.io/s/stale-closure-use-effect-broken-2-gyhzk )并点击几次增加按钮。然后看看控制台,每2秒出现一次Count is: 0,尽管count状态变量实际上已经增加了几次。9 N2 T/ w1 Q4 Q& f! V
( l' }$ w% c8 x; p6 i& h 为什么会这样?第一次渲染时,状态变量count初始化为0。组件安装后,useEffect()调用 setInterval(log, 2000)计时器函数,该计时器函数计划每2秒调用一次log()函数。在这里,闭包log()捕获到count变量为0。3 `6 G0 x* D" U
之后,即使在单击Increase按钮时count增加,计时器函数每2秒调用一次的log(),使用count的值仍然是0。log()成为一个过时的闭包。
8 i7 C L2 Y% ]7 F2 y" q& b 解决方案是让useEffect()知道闭包log()依赖于count,并在count改变时正确处理间隔的重置:
D% q8 W- P3 o4 p4 q function WatchCount() {
const [count, setCount] = useState(0);
useEffect(function() {
const id = setInterval(function log() {
console.log(`Count is: ${count}`);
}, 2000);
return function() {
clearInterval(id);
}
}, [count]);
return (
<div>
{count}
<button onClick={() => setCount(count + 1) }>
Increase
</button>
</div>
);
} 正确设置依赖项后,一旦count发生变化,useEffect()就会更新闭包。9 v: i& z# v0 L# e2 U) v
3.2 useState() ) S) O4 O) A$ H+ S1 f3 k& @
<DelayedCount>组件有1个button ,以1秒延迟异步增加计数器:( Y* S! x" w8 k0 w& Y5 d
function DelayedCount() {
const [count, setCount] = useState(0);
function handleClickAsync() {
setTimeout(function delay() {
setCount(count + 1);
}, 1000);
}
return (
<div> {count} <button onClick={handleClickAsync}>Increase async</button> </div>
);
} 现在打开演示(https://codesandbox.io/s/use-state-broken-0q994 )。快速单击2次按钮。计数器仅更新为1,而不是预期的2。
/ W0 m+ z Z& Z; L( s. C" \, d 每次单击setTimeout(delay, 1000)将在1秒后执行delay()。delay()此时捕获到的 count 为 0。两个delay()都将状态更新为相同的值:setCount(count + 1) = setCount(0 + 1) = setCount(1)。
* E" A3 J& U i! y- o" g 这是因为第二次单击的delay()闭包中已捕获了过时的count变量为0。为了解决这个问题,我们使用函数式方法setCount(count => count + 1)来更新count状态:
5 o- q% x4 u7 C1 U$ G function DelayedCount() {
const [count, setCount] = useState(0);
function handleClickAsync() {
setTimeout(function delay() {
setCount(count => count + 1); }, 1000);
}
function handleClickSync() {
setCount(count + 1);
}
return (
<div>
{count}
<button onClick={handleClickAsync}>Increase async</button>
<button onClick={handleClickSync}>Increase sync</button>
</div>
);
} 打开演示(https://codesandbox.io/s/use-state-fixed-zz78r )。再次快速单击按钮2次。计数器显示正确的值2。
" E5 U+ t8 G3 a& B( C 当一个返回基于前一个状态的新状态的回调函数被提供给状态更新函数时,React确保将最新的状态值作为该回调函数的参数提供:
; Z% v4 N- b3 m$ ]8 \ setCount(alwaysActualStateValue => newStateValue); 这就是为什么在状态更新过程中出现的过时装饰问题可以通过函数这种方式来解决。 H$ x* b" X j: U# S6 n
4.总结 $ M/ t# E+ z: G' m
当闭包捕获过时的变量时,就会发生过时的闭包问题。解决过时闭包的有效方法是正确设置React钩子的依赖项。或者,在失效状态的情况下,使用函数方式更新状态。% b2 g; Z1 O0 y* h5 O, O
C' ~) G( z9 }