最新公告
  • 欢迎您光临起源地模板网,本站秉承服务宗旨 履行“站长”责任,销售只是起点 服务永无止境!立即加入钻石VIP
  • useEffect钩子实现原理

    正文概述 掘金(隐冬)   2021-06-06   679

    useEffect这个钩子函数是用来模拟生命周期函数的。调用这个函数的时候他的第一个参数必须是一个函数,第二个参数可以不传也可以传递一个数组,当不传递的时候组件当中任何一个数据发生变化的时候useEffect这个钩子函数每次都会执行,当传递第二个参数数组的时候,数组中定义状态数据,当状态数据改变的时候再执行。

    useEffect(() => {
    
    })
    

    我们先来定义一下这个函数,这个函数接收两个参数。回调函数和依赖数组。

    我们首先要判断callback是不是函数,如果不是函数直接报错就可以了。

    function useEffect(callback, depsAry) {
        // 判断callback是否是函数
        if (Object.prototype.toString.call(callback) !== '[object Function]') {
            throw new Error('useEffect 函数的第一个参数必须是函数');
        }
    }
    

    接着需要判断depsAry是否传递,如果没有传递直接调用回调函数就可以了。

    // 判断depsAry有没有传递
    if (typeof depsAry === 'undefined') {
        callback();
    }
    

    如果depsAry传递了,我们需要判断是否是一个数组,如果不是就抛错。

    // 判断depsAry有没有传递
    if (typeof depsAry === 'undefined') {
        callback();
    } else {
        // 判断是否是数组
        if (Object.prototype.toString.call(depsAry) !== '[object Array]') {
            throw new Error('useEffect 函数的第二个参数必须是数组');
        }
    }
    

    如果传递的是一个数组,我们需要拿当前的依赖值和上一次的依赖值做对比,如果有变化就执行callback。

    // 存储上一次的依赖值
    let preDepsAry = [];
    // 判断是否是数组
    if (Object.prototype.toString.call(depsAry) !== '[object Array]') {
        throw new Error('useEffect 函数的第二个参数必须是数组');
    } else {
        // 将当前的依赖值和上一次的依赖值做对比, every如果返回true就是没变化,如果false就是有变化
        const hasChanged = depsAry.every((dep, index) => dep === preDepsAry[index]) === false;
        // 值如果有变化
        if (hasChanged) {
            callback();
        }
        // 同步依赖值
        preDepsAry = depsAry;
    }
    
    

    现在基本就写完了, 但是我们知道useEffect是可以多次调用的,我们这里存储的上一次的值只是最后一次的值,并不是每一次的。我们将preDepsAry变为一个二维数组,在数组中的每一个数组存储对应的depsAry。

    这里我们同样会用到索引。

    const preDepsAry = [];
    let effectIndex = 0;
    

    当我们对比的时候就不能直接使用preDepsAry对比了,需要做一些变化。

    if (Object.prototype.toString.call(depsAry) !== '[object Array]') {
        throw new Error('useEffect 函数的第二个参数必须是数组');
    } else {
        // 获取上一次的状态值
        const prevDeps = preDepsAry[effectIndex];
        // 如果存在就去做对比,如果不存在就是第一次执行
        // // 将当前的依赖值和上一次的依赖值做对比, every如果返回true就是没变化,如果false就是有变化
        const hasChanged = prevDeps ? depsAry.every((dep, index) => dep === prevDeps[index]) === false : true;
        // 值如果有变化
        if (hasChanged) {
            callback();
        }
        // 同步依赖值
        preDepsAry[effectIndex] = depsAry;
    }
    

    多次调用的时候需要让effectIndex加1

    // 同步依赖值
    preDepsAry[effectIndex] = depsAry;
    effectIndex++;
    

    这里我们不能让effectIndex一直加,需要在组件重新渲染的时候恢复成0,之前的render函数中归零就可以了。

    function render() {
        stateIndex = 0;
        effectIndex = 0;
        ReactDOM.render(<App />, document.getElementById('root'));
    }
    

    这样我们就写完了。逻辑还是比较简单的。

    // 重新渲染函数
    function render() {
        stateIndex = 0;
        effectIndex = 0;
        ReactDOM.render(<App />, document.getElementById('root'));
    }
    
    // 存储上一次的依赖值
    const preDepsAry = [];
    let effectIndex = 0;
    function useEffect(callback, depsAry) {
        // 判断callback是否是函数
        if (Object.prototype.toString.call(depsAry) !== '[object Array]') {
            throw new Error('useEffect 函数的第二个参数必须是数组');
        } else {
            // 获取上一次的状态值
            const prevDeps = preDepsAry[effectIndex];
            // 如果存在就去做对比,如果不存在就是第一次执行
            // // 将当前的依赖值和上一次的依赖值做对比, every如果返回true就是没变化,如果false就是有变化
            const hasChanged = prevDeps ? depsAry.every((dep, index) => dep === prevDeps[index]) === false : true;
            // 值如果有变化
            if (hasChanged) {
                callback();
            }
            // 同步依赖值
            preDepsAry[effectIndex] = depsAry;
            // 累加
            effectIndex++;
        }
    }
    

    起源地下载网 » useEffect钩子实现原理

    常见问题FAQ

    免费下载或者VIP会员专享资源能否直接商用?
    本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。
    提示下载完但解压或打开不了?
    最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,若小于网盘提示的容量则是这个原因。这是浏览器下载的bug,建议用百度网盘软件或迅雷下载。若排除这种情况,可在对应资源底部留言,或 联络我们.。
    找不到素材资源介绍文章里的示例图片?
    对于PPT,KEY,Mockups,APP,网页模版等类型的素材,文章内用于介绍的图片通常并不包含在对应可供下载素材包内。这些相关商业图片需另外购买,且本站不负责(也没有办法)找到出处。 同样地一些字体文件也是这种情况,但部分素材会在素材包内有一份字体下载链接清单。
    模板不会安装或需要功能定制以及二次开发?
    请QQ联系我们

    发表评论

    还没有评论,快来抢沙发吧!

    如需帝国cms功能定制以及二次开发请联系我们

    联系作者

    请选择支付方式

    ×
    迅虎支付宝
    迅虎微信
    支付宝当面付
    余额支付
    ×
    微信扫码支付 0 元