【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的其它逻辑 

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

相关推荐
meilindehuzi_a2 分钟前
深入理解 JavaScript 的同步与异步机制:从单线程设计到 Promise 核心应用
开发语言·javascript·ecmascript
如烟花的信页5 分钟前
加速乐cookie逆向分析
javascript·爬虫·python·js逆向
永远的WEB小白10 分钟前
css改变svg图标的颜色
前端·javascript·css
ikoala29 分钟前
Codex 不得不装的 12 个插件,都在这了
前端·javascript·后端
赵庆明老师1 小时前
JS检查提交的文件是否合规
开发语言·前端·javascript
颂love1 小时前
Vue的两大生态以及组件通信
前端·javascript·vue.js·typescript
光影少年2 小时前
js单线程,为什在node环境下的js可以处理高并发请求?
前端·javascript·掘金·金石计划
vim怎么退出2 小时前
Dive into React——事件系统
前端·react.js·源码阅读
moMo2 小时前
# JavaScript 的“等等我”:聊聊同步与异步
javascript
Cobyte2 小时前
19.Vue Vapor 的实现原理原来这么简单
前端·javascript·vue.js