JavaScript闭包

目录

作用域

作用域链

闭包 

1、什么是闭包

2、闭包产生的条件

3、闭包的创建

3、闭包的用途

4、闭包使用的注意事项

5、闭包中的this指向


作用域

ES5中,作用域主要分为局部作用域和全局作用域。

局部作用域:也称为函数作用域,在函数内部声明的变量属于局部变量,且函数内部声明的变量只能在函数内部引用,不能在函数外部访问到。

全局作用域:在函数外声明的变量叫做全局变量,全局变量不仅可以在函数外部使用,还能在函数内部引用。

var a = 10;//全局变量
function foo() {
    console.log(a);
    var a = 100;//局部变量
    console.log(a); 
}
foo()//结果输出:undefined 100

//全局变量可以在函数内部使用
var a = 10;
function foo() {
    console.log(a);
}
foo()//结果输出:10

//函数内部的变量不能在函数外部访问到
var a
function foo() {
    console.log(a);
    var a = 100;
    console.log(a); 
}
foo()
console.log(a);//结果输出:undefined 100 undefined

 由上面的案例可以看出作用域的作用之一就是隔离变量,即使变量名相同,但是作用域不同,因此不会在使用变量的过程中产生冲突。

作用域链

当自由变量顺着作用域寻找时,如果其父级作用域和祖级作用域都有自由变量的值,此时自由变量将优先选择父级作用域的值,遵循就近原则。 

闭包 

1、什么是闭包

闭包是指有权访问另一个函数作用域中的变量的函数,能够读取其他函数内部变量的函数。在JavaScript中读取函数的局部变量只能通过该函数的子函数。在一个函数内部嵌套另一个函数,内部函数引用了外部函数的数据形成一个闭包,内部的函数可以理解为外部函数的闭包函数。

2、闭包产生的条件

(1)函数嵌套函数

(2)内部函数应用了外部函数的数据

(3)参数和变量不会被销毁

(4)一个函数return了另外一个函数。(这个条件在网上的说法不一,有些人认同,有些人不认同)

3、闭包的创建

创建闭包最常见的方式就是函数嵌套函数

function foo() {
  var a = 1;
  var b = 2;
  function fn() {
    return a + b;
  }
  return fn;
}
console.log(foo()()); // 3

3、闭包的用途

(1)可以访问到内部函数的变量

(2)可以保持变量在内存中不被销毁

function f1() {
  var n = 999;
  nAdd = function () { n += 1 }
  function f2() {
    console.log(n);
  }
  return f2;
}
var result = f1(); // 相当于f2()
result(); // 999
nAdd();
result(); // 1000 函数f1中的局部变量n一直保存在内存中,并没有在f1调用后被自动清除

在上面的案例中,f1是f2的父函数,而f2被赋给了一个全局变量,这导致f2始终在内存中,而f2的存在依赖于f1,因此f1也始终在内存中,不会在调用结束后,被垃圾回收机制回收。

注意"nAdd=function(){n+=1}"这一行,首先在nAdd前面没有使用var关键字,因此nAdd是一个全局变量。其次,nAdd的值是一个匿名函数,而这个匿名函数本身也是一个闭包,所以nAdd相当于是一个setter,可以在函数外部对函数内部的局部变量进行操作。

4、闭包使用的注意事项

(1)由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在IE中可能导致内存泄露,这是IE的BUG。

解决方法:在退出函数之前,将不使用的局部变量全部删除。

(2)闭包会在父函数外部,改变父函数内部变量的值。所以,如果你把父函数当作对象使用,把闭包当作它的公用方法,把内部变量当作它的私有属性,一定不要随便改变父函数内部变量的值。多个子函数的scope都是同时指向父级,是完全共享的。因此当父级的变量对象被修改时,所有子函数都受到影响。

5、闭包中的this指向

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>Document</title>
</head>
<body>
  <script>
    var name = 'window'
    var obj = {
      name: 'obj',
      say: function () {
        return function () {
          console.log(this.name);
        }
      }
    }
    var x = obj.say()
    x() 
  </script>
</body>
</html>

控制台输出的结果为window 。因为此时obj.say()被赋值给了一个全局变量x,即是say()这个子函数在全局中被调用了,谁调用this就指向谁,所以此时this指向全局,因此this.name此时应为window。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值