学习记录:使用立即调用的函数表达式创建局部作用域

这段程序(bug程序)输出什么?

  function wrapElements(a) {
        var result = [], i, n;
        for (i = 0, n = a.length; i < n; i++) {
           result[i] = function () { return a[i]; };
        }
        return result
    }
    var wrap = wrapElements([10, 20, 30, 40, 50])
    var f = wrap[0]
    f();//?

你可能希望这段程序输出10,但实际上它输出的是 undefined 值

这是因为:我们希望该函数存储的是嵌套函数创建时变量 i 的值。但事实上,它存储的是变量 i 的引用。由于每次函数创建后变量 i 的值都发生了变化,因此内部函数在执行时拿到的是变量 i 的最终值(5)。值得注意的是,闭包存储的是外部变量的引用而不是值。

所以,我们调用其中任何一个闭包,它都会查找数组的索引 5 并返回 undefined 值。解决方法是通过创建一个嵌套函数,并立即调用它来强制创建一个局部作用域

    function wrapElements(a) {
        var result = [], i, n;
        for (i = 0, n = a.length; i < n; i++) {
            (function () {
                var j = i;
                result[i] = function () { return a[j]; };
            })();
        }
        return result
    }
    var wrap = wrapElements([10, 20, 30, 40, 50])
    var f = wrap[0]
    f();//10

这种技术被称为立即调用的函数表达式,或IIFE。它是一种不可或缺的解决JavaScript缺少块级作用域的方法。另一种变种是将作为形参的局部变量绑定到IIFE并将其值作为实参传入。

  function wrapElements(a) {
        var result = [], i, n;
        for (i = 0, n = a.length; i < n; i++) {
            (function (j) {
                result[i] = function () { return a[j]; };
            })(i)
        }
        return result
    }
    var wrap = wrapElements([10, 20, 30, 40, 50])
    var f = wrap[0]
    f();//10
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值