`

JavaScript Garden - 闭包和引用

阅读更多

闭包和引用

闭包是 JavaScript 一个非常重要的特性,这意味着当前作用域总是能够访问外部作用域中的变量。
因为 函数 是 JavaScript 中唯一拥有自身作用域的结构,因此闭包的创建依赖于函数。

模拟私有变量

function Counter(start) {
    var count = start;
    return {
        increment: function() {
            count++;
        },

        get: function() {
            return count;
        }
    }
}

var foo = Counter(4);
foo.increment();
foo.get(); // 5

这里,Counter 函数返回两个闭包,函数 increment 和函数 get。 这两个函数都维持着
对外部作用域 Counter 的引用,因此总可以访问此作用域内定义的变量 count.

为什么不可以在外部访问私有变量

因为 JavaScript 中不可以对作用域进行引用或赋值,因此没有办法在外部访问 count 变量。
唯一的途径就是通过那两个闭包。

var foo = new Counter(4);
foo.hack = function() {
    count = 1337;
};

上面的代码不会改变定义在 Counter 作用域中的 count 变量的值,因为 foo.hack 没有
定义在那个作用域内。它将会创建或者覆盖全局变量 count

循环中的闭包

一个常见的错误出现在循环中使用闭包,假设我们需要在每次循环中调用循环序号

for(var i = 0; i < 10; i++) {
    setTimeout(function() {
        console.log(i);  
    }, 1000);
}

上面的代码不会输出数字 0 到 9,而是会输出数字 10 十次。

当 console.log 被调用的时候,匿名函数保持对外部变量 i 的引用,此时 for循环已经结束,i 的值被修改成了 10.

为了得到想要的结果,需要在每次循环中创建变量 i 的拷贝

避免引用错误

为了正确的获得循环序号,最好使用 匿名包裹器译者注其实就是我们通常说的自执行匿名函数)。

for(var i = 0; i < 10; i++) {
    (function(e) {
        setTimeout(function() {
            console.log(e);  
        }, 1000);
    })(i);
}

外部的匿名函数会立即执行,并把 i 作为它的参数,此时函数内 e 变量就拥有了 i 的一个拷贝。

当传递给 setTimeout 的匿名函数执行时,它就拥有了对 e 的引用,而这个值是不会被循环改变的。

有另一个方法完成同样的工作;那就是从匿名包装器中返回一个函数。这和上面的代码效果一样。

for(var i = 0; i < 10; i++) {
    setTimeout((function(e) {
        return function() {
            console.log(e);
        }
    })(i), 1000)
}

分享到:
评论

相关推荐

    JavaScript_Garden_CN

    2. **函数**: 包括函数声明与表达式、`this`的工作原理、闭包与引用、`arguments`对象、构造函数、作用域与命名空间等内容,深入探讨了函数的灵活性和复杂性。 3. **数组**: 阐述了数组的遍历与属性、`Array`构造...

    Javascript Garden 网页的静态文件

    JavaScript Garden,通常被称为JS Garden,是一个在线资源,旨在帮助开发者理解和规避JavaScript中的常见陷阱和微妙之处。这个项目是由Bodo Tasche创建的,它详细解释了JavaScript语言的一些复杂性和不直观的行为,...

    javascript secrit garden

    《JavaScript秘密花园》是一本专注于JavaScript语言中容易让人迷惑或者陷阱重重的特性的...通过阅读《JavaScript秘密花园》这样的资源,开发者可以更好地理解和避免JavaScript中的常见陷阱,提高代码质量和可维护性。

    javascript_garden:javascript后花园-算法之路

    文件名"javascript_garden-master"可能代表这是一个开源项目,"master"分支通常表示项目的主分支,其中包含了项目的核心代码和文档。用户可以通过下载和阅读这个资源来学习JavaScript的高级概念,以及如何将这些概念...

    pratiknilange.github.io:回家! :house_with_garden:

    2. **函数**:声明函数、匿名函数、箭头函数、作用域和闭包。 3. **对象和数组**:对象字面量、构造函数、原型链、数组方法(如map、filter、reduce等)。 4. **DOM操作**:获取元素、添加/删除元素、事件处理。 5. ...

Global site tag (gtag.js) - Google Analytics