Javascript 闭包的理解

 示例1:

function makeCounter() {
    var i = 0;
    console.log( ++i );
}

var counter = makeCounter(); // 输出: 1
counter();  //TypeError: undefined is not a function
counter();  //TypeError: undefined is not a function

这个例子中,声明了一个makeCounter函数,发现 var counter = makeCounter(); 输出了一个结果: 1   这是因为实际上  赋值号的右边 makeCounter() 就把函数执行了一遍,但没有返回值,所以 counter 被赋予了 undefined ,接下来的两次调用返回错误 TypeError: undefined is not a function

 

再来看修改后的代码:

示例2:

复制代码
function makeCounter() {
    var i = 0;
    return function() {
        console.log( ++i );
    };
}

var counter = makeCounter;  //无输出
counter(); // 无输出
counter(); // 无输出

var counter2 = counter();  //无输出
counter2(); // 输出: 1
counter2(); // 输出: 2

var counter3 = makeCounter();  //无输出
counter3(); // 输出: 1
counter3(); // 输出: 2

console.log(i);
// ReferenceError: i is not defined
复制代码

这次 makeCounter函数返回值是一个函数,这里其实是一个闭包,根据javascript高级程序设计,闭包是指有权访问另一个函数作用域中的变量的函数。 根据 MDN:

Closures are functions that refer to independent (free) variables.

In other words, the function defined in the closure 'remembers' the environment in which it was created.

指通常我们把像这样在一个函数内部再创建一个函数就是创建闭包的最常见方式。

闭包中的函数能够“记住”创建它的外层环境的活动对象,makeCount函数执行之后,其活动对象并不会被销毁,因为闭包内的函数的作用域链仍然在引用这个活动对象,因此仍然保留在内存中。从垃圾回收角度看就是外部执行环境的活动对象不会被当做垃圾回收。

如上面代码所示,值得注意的是每次调用makeCount函数,即每次返回闭包时虽然内部函数体是一样的,但是每次执行一次,它执行环境的活动对象都是独立的,互相并不关联,请见 count2() 和count3() 的输出。

可以说 javascript中的函数都在闭包中,因为每个函数都能够访问到函数体之外的全局执行环境。

 

下一篇:匿名函数与闭包

posted @   Tong Zeng  阅读(229)  评论(0编辑  收藏  举报
编辑推荐:
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
阅读排行:
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· DeepSeek如何颠覆传统软件测试?测试工程师会被淘汰吗?
点击右上角即可分享
微信分享提示