【React中的闭包陷阱】

闭包陷阱

前言

以下分析按照react的函数组件写法进行

什么是闭包

函数能够访问并记住其定义时所在作用域(即使该函数在定义的作用域外执行),这个函数 + 词法环境的整体就是闭包

typescript 复制代码
// 节流实现 - 闭包
function throttle(fn, dealy) {
	let timerId; // 定时任务id
	let targetArgs; // 存储尾参数
	return function (...args) {
		targetArgs = args;
		if (!timerId) {
			timerId = setTimeout(() => {
				fn.apply(this, targetArgs)
				timerId = null
			}, dealy)
		}
	}
}
function throttleTarget(a, b) {
	return a + b;
}
const throttleEffect = throttle(throttleTarget, 500) // throttleEffect从throttle中返回,持有词法作用域变量timerId和targetArgs的引用,进而形成闭包

为什么react会存在闭包陷阱

首先react在update的时候会重新执行Component(组件函数),进而生成新的词法作用域的变量。假如你存在会锁住某一过程中词法作用域的逻辑,就会产生闭包陷阱的问题。

typescript 复制代码
// 子组件
import React, { useState, forwardRef, Ref, useImperativeHandle, useEffect, useCallback } from "react";

export interface ItemRef {
	queryData: () => void
}

const Item = forwardRef((props: { value: number }, ref: Ref<ItemRef>) => {
	const [ value, setValue ] = useState(0)
	const queryData = () => {
		console.log("触发了防抖的函数", props, isEffect && "useEffect进入");
		setValue(props.value * 2);
	}
	const queryDataCallback = useCallback(queryData, []) // 因为只会在mountd的时候执行一次,所以queryDataCallback所持有的词法作用域永远是第一次的,所以调用queryDataCallback所触发的queryData所使用的词法作用域也理所应当是第一次的,所以props.value永远是0
	useEffect(() => {
		setInterval(() => {
			queryDataCallback(true);
		}, 1000)
	}, [])
	useImperativeHandle(ref, () => {
		return queryData // 同理,因为只会初始化的时候执行一次,所以保存的queryData的指针永远是第一次执行时生成的,因此保存的词法作用域也是第一次,因此执行的时候props.value也是0
	}, [])
})

// 父组件
import React, { useState, useRef, useEffect } from "react";
const Closure = () => {
    const [ count, setCount ] = useState(0);
    const itemRef = useRef<ItemRef>(null);
    const triggerControll = () => {
        setCount(count + 1);
    };
    useEffect(() => {
        itemRef.current?.queryData();
    }, [count]);
    return (
        <div>
            <button onClick={ triggerControll }>点击次数:{count}</button>
            <Item ref={ itemRef } value={count}></Item>
        </div>
    );
};

props在更新的时候也会生成一个新的指针指向新的对象,而非在原有对象上进行的修改,所以如果这里按照指针不变来获取最新的值的话,就会遇到隐藏的闭包陷阱的问题。

如何解决闭包陷阱

要避免闭包陷阱的产生,需要时刻注意所使用的词法作用域是不是旧的。上述两种情况处理的方式尽均可以修改成监听props.value的变化来生成信息的值

typescript 复制代码
// ... 此处省略Item的其它逻辑
const queryDataCallback = useCallback(queryData, [props.value]);
useEffect(() => {
    setInterval(() => {
        queryDataCallback(true);
    }, 1000);
}, [props.value]);
useImperativeHandle(ref, () => {
    return {
        queryData
    };
}, [props.value]);
// 此处省略Item的其它逻辑 

这样就不会出现闭包陷阱的情况了

相关推荐
山河木马7 小时前
矩阵专题3-怎么创建投影矩阵(uProjectionMatrix)
javascript·webgl·计算机图形学
泯泷8 小时前
第 2 篇:设计第一套字节码:Opcode、Instruction 与 Constant Pool
前端·javascript·安全
泯泷8 小时前
第 1 篇:从 1 + 2 开始:亲手写出第一台 JSVM
前端·javascript·安全
朦胧之9 小时前
页面白屏卡住排查方法
前端·javascript
Ruihong9 小时前
Vue withDefaults 转 React:VuReact 怎么处理?
vue.js·react.js·面试
犇驫聊AI10 小时前
Chrome DevTools MCP + Claude Code 自定义skills生成接口代码生成器
前端·javascript
kyriewen10 小时前
别再这样写 async/await 了:我在 Code Review 中见过最多的 8 个错误
前端·javascript·面试
用户2986985301415 小时前
在 React 中使用 JavaScript 将 Excel 转换为 SVG
前端·javascript·react.js
labixiong15 小时前
手写Promise--微任务、静态方法、async/await 全搞懂(三)
前端·javascript
铁皮饭盒17 小时前
3行代码搞定页面截图,Bun.WebView真的简单
javascript