闭包

本文详细解释了闭包的概念及其工作原理,并探讨了闭包可能导致的内存泄漏问题。通过具体示例展示了如何创建闭包,以及闭包如何保持对外部变量的引用,从而可能导致内存泄漏。最后提供了解决内存泄漏的方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

闭包是指有权访问另一个函数作用域中变量的函数。创建闭包的常见方式,就是在一个函数内部创建另一个函数。

无论什么时候在函数中访问一个变量时,就会从作用域链中搜索具有相应名字的变量。一般来讲,当函数执行完毕后,局部活动对象就会被销毁,内存中仅保存全局作用域(全局执行环境的变量对象)。但是闭包的情况又有所不同,由于闭包会携带包含它的作用域,因此会比其他函数占用更多的内存。

如下为一个闭包的实例:

function init() {
    var name = "Mozilla"; // name 是一个被 init 创建的局部变量
    function displayName() { // displayName() 是内部函数,一个闭包
        alert(name); // 使用了父函数中声明的变量
    }
    displayName();
}
init();

init() 创建了一个局部变量 name 和一个名为 displayName() 的函数。displayName() 是定义在 init() 里的内部函数,仅在该函数体内可用。displayName() 内没有自己的局部变量,然而它可以访问到外部函数的变量,所以 displayName() 可以使用父函数 init() 中声明的变量 name 。但是,如果有同名变量 name 在 displayName() 中被定义,则会使用 displayName() 中定义的 name 。

因此闭包具有以下三个特点

1、函数内部嵌套函数;

2、内部函数可以访问外部函数的变量;

3、闭包的参数和变量不会被回收。

闭包的作用:

设计私有变量及方法

闭包的缺点:

常驻内存,会增大内存使用量,使用不当很容易造成内存泄露。

导致的内存泄漏

循环引用

闭包可能会导致在不经意间创建循环引用。因为函数是必须保存在内存中的对象,所以位于函数执行上下文中的所有变量也需要保存在内存中:

function outerFn() {
    var outerVar = {};
    function innerFn() {
        console.log(outerVar);
    }
    outerVar.fn = innerFn;
    return innerFn;
};

这里创建了一个名为 outerVar 的对象,该对象在内部函数innerFn()中被引用。然后,为 outerVar 创建了一个指向 innerFn()的属性,之后返回了innerFn()。这样就在 innerFn() 上创建了一个引用outerVar的闭包,而outerVar又引用了innerFn()。

这会导致变量在内存中存在的时间比想象得长,而且又不容易被发现。这还不算完,还有可能会出现比这种情况更隐蔽的引用循环:

function outerFn() {
    var outerVar = {};
    function innerFn() {
        console.log('hello');
    }
    outerVar.fn = innerFn;
    return innerFn;
};

这里我们修改了innerFn(),不再招惹 outerVar。但是,这样做仍然没有断开循环引用。
即使innerFn()不再勾引 outerVar,outerVar 也仍然位于innerFn()的封闭环境中。由于闭包的原因,位于 outerFn()中的所有变量都隐含地被 innerFn()所引用。我们再想一想,在 java 中的内部类不也是类似当前情况吗,内部类能够‘看’外部的 this。此时此刻,正如彼时彼刻,竟如此相像。因此,闭包会使意外地创建这些引用循环变得易如反掌。

DOM与JavaScript的循环

闭包实际上非常容易造成JavaScript对象和DOM对象的隐蔽循环引用。来看看下面的例子:

  1. function example(){

  2. var element =document.getElementByID("div1"); //①

  3. element.onclick = function() {

  4. alert("This is a leak!"); //②

  5. }

  6. }

以上函数example() 中用匿名函数创建了一个闭包。

第①句:JavaScript对象element引用了一个DOM对象(其id为“div1”); JS(element) ----> DOM(div1)

第②句:该DOM对象的onclick属性引用了匿名函数闭包,而闭包可以引用外部函数example() 的整个活动对象,包括elemnt ; DOM(div1.onclick) ---->JS(element)
由此形成了JavaScript对象和DOM对象的隐蔽循环引用。

 2.解决方法:

常用的解决方法就是在JavaScript代码段运行完之时将形成循环引用的JavaScript对象手动设置为空,切断引用。

修改的例子如下:

 


 
  1. function example(){

  2. var element =document.getElementByID("div1"); //①

  3. element.onclick = function() {

  4. alert("This is a leak!"); //②

  5. }

  6. element = null; //添加的语句

  7. }


造成内存泄露的原因还有很多,而解决方法也很多

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值