JS闭包

作者: cs0710 | 来源:发表于2019-07-01 15:41 被阅读0次

闭包:在自己的作用域中引用了外层作用域变量的函数。
这里有两个关键词:1.外层作用域变量 2.函数


简单闭包

// 日常开发中最常见的闭包case
let variable = 1;
function fn(){
    console.log(variable);
}
fn();  // 1

闭包的局限

// 这是全局作用域, 相对于两个函数作用域来说是全局作用域就是外层作用域
// 在外层作用域中定义一个变量
let outerVariable = 1;

function fn(){
    // 这里面是函数作用域, 相对说也是内层作用域
    // 在内层作用域内定义一个变量
    let innerVariable = 2;

    // 在内层作用域内访问外层作用域中的变量 outerVariable
    console.log(outerVariable); // 1
}

fn(); // 调用函数, 输出 1, 说明在函数(内层)作用域里可以访问到全局(外层)作用域里的变量

// 在外层作用域里尝试访问内层作用域的变量
console.log(innerVariable); // ReferenceError: innerVariable is not defined
// 报了引用错误, 说明外层作用域中没有 innerVariable 这个变量, 也就是说明外层无法访问到内层作用域中定义的变量。

通过闭包实例,访问引用的外层作用域变量

function outer(){
  let outerVariable = 0;
  let inner = function(){
    console.log('outerVariable === ', outerVariable);
  }

  return inner;
}
  
let closure = outer(); // 调用 outer 函数, 得到 inner 函数, 也就是得到了一个闭包实例

// 运行这个闭包, 在这个闭包里仍然可以访问它引用的外层作用域里的变量 outerVariable
closure(); // outerVariable ===  0

闭包实例的环境互相独立

function outer(){
    let outerVariable = 0;

    let inner = function(){
        outerVariable ++; // 增加的代码, 在输出 outerVariable 之前, 先让他自增 1
        console.log('outerVariable === ', outerVariable);
    }
    return inner;
}

let closure1 = outer(); // 修改的代码, 实例化一个闭包并命名为 closure1
let closure2 = outer(); // 修改的代码, 再实例化第二个闭包并命名为 closure2

// 增加的代码, 调用三次 closure1, 观察 closure1 引用的变量 outerVariable 的变化
closure1();  // outerVariable ===  1
closure1();  // outerVariable ===  2
closure1();  // outerVariable ===  3

// 增加的代码, 调用一次 closure2, 观察 closure2 引用的变量 outerVariable 的值
closure2();  // outerVariable ===  1
/* 发现闭包 closure2 引用的 outerVariable 值没有受到闭包 closure1 的影响*/

经典闭包,考查闭包实例环境的独立性

function Person(){
    let age = 24;

    this.getAge = function(){
        return age;  // 引用了外层的变量
    }

    this.grow = function(){
        age ++;  // 引用了外层的变量
    }
}

let xm = new Person(); // 实例化小明 xm 
// 无法通过 xm.age 来访问 xm 的年龄;

// 获取 xm 的年龄
let ageThisYear = xm.getAge();
console.log('小明今年的年龄是 ' + ageThisYear); // 小明今年的年龄是 24

// 过了一年, xm 涨了一岁
xm.grow();

// 获取小明过年之后的年龄
let ageNextYear = xm.getAge();
console.log('小明明年的年龄是', ageNextYear); // 小明今年的年龄是 24

相关文章

  • php之闭包函数(Closure)

    php闭包函数(Closure) JS闭包 js和php闭包使用和区别

  • JS闭包

    JS闭包 闭包练习

  • JS闭包问题(二)

    在之前的JS闭包问题(一)文章中大概介绍了一下JS闭包,同时讲了闭包与变量之间的问题,今天我们继续聊闭包,聊聊闭包...

  • JS闭包大结局(JS闭包系列3)

    在上一篇中再谈JS闭包(JS闭包系列2),我详细的介绍了JS中的变量作用域相关的概念,结合第一节关于JS闭包(JS...

  • JS闭包入门

    最近有看到朋友的面经里提到的JS闭包的问题,就想研究研究,以下是我对JS闭包的简单理解。 到底什么是JS闭包? 定...

  • 学习JavaScript闭包和作用域笔记

    JS JavaScript闭包和作用域 闭包 JavaScript高级程序设计中对闭包的定义:闭包是指有权访问另外...

  • 再谈JS闭包(JS闭包系列2)

    这篇文章,来继续谈谈Javascript闭包的剩余问题。因为在上一篇文章中关于JS闭包(JS闭包系列1)主要简单的...

  • 简单的聊一下闭包

    js中的闭包 闭包是学习js中永远也绕不过去的一个坎,那么,今天我们就去一段简单的代码开始聊一聊闭包 什么是闭包 ...

  • 浅谈闭包

    js中的闭包 闭包是学习js中永远也绕不过去的一个坎,那么,今天我们就去一段简单的代码开始聊一聊闭包 什么是闭包 ...

  • 闭包??

    什么闭包,闭包有什么用?http://js.jirengu.com/pogadikofa/1/闭包是在某个作用域内...

网友评论

    本文标题:JS闭包

    本文链接:https://www.haomeiwen.com/subject/yqogcctx.html