Loading [MathJax]/jax/output/CommonHTML/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >javaScript核心技术--“闭包”,不看绝对后悔!

javaScript核心技术--“闭包”,不看绝对后悔!

作者头像
Java深度编程
发布于 2020-06-10 06:51:03
发布于 2020-06-10 06:51:03
42200
代码可运行
举报
文章被收录于专栏:Java深度编程Java深度编程
运行总次数:0
代码可运行

“闭包”,又称“定义在函数内部的函数”,闭包技术是javaScript中很关键的核心技术,很多框架的研发或者企业高端技术都需要使用到它。要理解闭包技术,必须先弄明白“变量的作用域”。 1.变量的作用域 javaScript沿袭的java的变量规则,但稍有改进。和java一样可分为“全局变量”和“局部变量”,在javaScript中的“局部变量”又称之为函数变量。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制


    var x = 999;
 
    function f1() {
      var y = 888;
      console.log(x); 
      console.log(y); 
    }
    f1() // 999  888
    console.log(y); // Uncaught ReferenceError: n is not defined

由上可知javaScript和java一样,父对象的所有变量,对子对象都是可见的,反之则不成立。而有些时候需要在外部父对象中获取子对象区域内部的变量,正常情况下是无法做到的,这时候就需要用到“闭包”技术了。

2.什么是闭包 请先看以下的函数:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制


    function f1() {
      var n = 999;
      function f2() {
      console.log(n); // 999
      }
    }

上面代码中,函数f2就在函数f1内部,这时f1内部的所有局部变量,对f2都是可见的。但是反过来就不行,f2内部的局部变量,对f1就是不可见的。这就是 JavaScript 语言特有的"链式作用域"结构(chain scope),子对象会一级一级地向上寻找所有父对象的变量。所以,父对象的所有变量,对子对象都是可见的,反之则不成立。上面的代码中内部函数分f2()就是“闭包”,一个定义在函数内部的函数。 3.闭包的用途 3.1. 突破局域限制,读取函数内部的变量值。 逻辑思维分析: 上面我们已经知道了函数f2()就是闭包,那么我们如果去使用它获取函数内部的变量呢? 分析:既然f2可以读取f1的局部变量,那么只要把f2作为返回值,我们不就可以在外部得到返回值,进而间接读取它的内部变量了吗!

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制


    function f1() {
      var n = 999;
      function f2() {
        console.log(n);
      }
      return f2;
    }
 
    var result = f1();
    result(); // 999

上面代码中,函数f1的返回值就是函数f2,由于f2可以读取f1的内部变量,所以就可以在外部获得f1的内部变量了。

3.2.“记住”诞生的环境 闭包最大的特点,就是它可以“记住”诞生的环境,比如f2记住了它诞生的环境f1,所以从f2可以得到f1的内部变量。在本质上,闭包就是将函数内部和函数外部连接起来的一座桥梁。它可以让这些变量始终保持在内存中,使得它诞生环境一直存在。 现在,假设我们有一个需求:每调用一次函数,都记录这个函数的被调用的次数。如何实现?用我们常规的思维,肯定是定义一个外部变量,然后每调用一次就++,如下所示:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制


    var start = 0;
    function test_01() {
      start++;
    }
 
    test_01();
    start // 1
    test_01();
    start // 2
    test_01();
    start // 3

你们会发现,上面的方式完美的实现了。但假如需求在改动一下,函数test_01()内部还有一个函数test_02(),要录test_02()函数被调用的次数,这个时候如何实现呢?我们继续按上面的套路搬砖:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制


    var start = 0;
    function test_01() {
 
        function test_02() {
            start++;
        }
    }
 
    test_01();
    start // 0
    test_01();
    start // 0
    test_01();
    start // 0

这时候你们会发现,无论你调用多少次函数,start都不会增长,一直是0。why? 但如果你把上面的代码改一改,将函数test_02作为返回值,并且外部定义一个变量接受它,就不一样了。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制


    var start = 0;
    function test_01() {
 
       return function test_02() {
            start++;
        }
    }
 
    var temp = test_01();
    temp ();
    start // 1
    temp ();
    start // 2
    temp ();
    start // 3

可以看到start的值又神奇般的增长了。这究竟是为什么呢?你是否感觉到了想破脑袋也想不明白是为什么?哈哈…… 其实这就是闭包技术的一种体现。用比较科学的术语来技术就是:“temp始终在内存中,而temp的存在依赖于函数test_01(),函数test_01()也因此始终在内存中,不会在调用结束后,被垃圾回收机制回收。所以它才能一直记录下这个‘诞生环境’ ”。 上面的这种解释可能过于“科学语言”,让人难以理解。因此我用比较通俗的语言来解释:因为我在外部声明了一个变量temp,它调用了函数test_01(),而test_01()又返回了函数test_02()。所以上面的代码可以等价与下面的代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制


    var start = 0;
    function test_01() {
 
       return function test_02() {
            start++;
        }
    }
 
    var temp = function test_02() {
            start++;
        };
    temp ();
    start // 1
    temp ();
    start // 2
    temp ();
    start // 3

上面我举的第一个例子就已经很好的说明了,在同一个作用域操作一个变量是可以成功的。这种变换操作手法更this的作用域极其相似。javaScript中this始终指向当前对象,然而this的指向却是动态的。说到这里了我就随便提一提this的作用域吧。 this的作用域

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制


 
    var A = {
      name: '张三',
      describe: function () {
        return '姓名:'+ this.name;
      }
    };
 
    var B = {
      name: '李四'
    };
 
    B.describe = A.describe;
    B.describe()
    // "姓名:李四"
 
    var name = '龚文学';
    var tenp = function () {
        return '姓名:'+ this.name;
      };
 
    temp(); // 姓名:龚文学

从上面可以看出在A对象和B对象调用同一个函数this的指向不同,所以输出了不能的结果。如果把这个函数提取出来,赋值给一个变量,this的指向就是最顶层的window对象,这个时候就输出了我的顶顶大名--“龚文学”。这与“闭包”的方式十分类似,我以此举例说明,希望能帮助大家理解。如果大家还是有不懂的地方,请在微信公众平台《Java深度编程》留言。

3.3.封装对象的私有属性和私有方法

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制


    function Person(name) {
      var _age;
      function setAge(n) {
        _age = n;
      }
      function getAge() {
        return _age;
      }
 
      return {
        name: name,
        getAge: getAge,
        setAge: setAge
      };
    }
 
    var p1 = Person('张三');
    p1.setAge(25);
    p1.getAge() // 25

上面代码中,函数Person的内部变量_age,通过闭包getAge和setAge,变成了返回对象p1的私有变量。因为闭包能一直记住之前的环境,所以Person的内部变量会随之永久改变,这与java的get,set方式十分类似。 4.闭包的弊端 注意,因为外层函数每次运行,都会生成一个新的闭包,而这个闭包又会保留外层函数的内部变量,外层函数多次运行后会导致内存消耗很大。因此不能滥用闭包,否则会造成网页的性能问题。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2018-10-18,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Java深度编程 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Javascript闭包剖析(通俗易懂)
闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现。 下面就是我的学习笔记,对于Javascript初学者应该是很有用的。 一、变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域。 变量的作用域无非就是两种:全局变量和局部变量。 Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量。 var n=999; function f1(){ alert(n); } f1()
Marser
2018/06/25
4330
JavaScript闭包(Closure)
上面的代码中,函数 f2 就被包括在函数 f1 内部,这时 f1 内部的所有局部变量,对 f2 都是可见的。
Leophen
2019/08/23
6210
🌞 深入剖析 JavaScript 闭包
🌞 深入剖析 JavaScript 闭包 💎导读目录 什么是闭包 闭包的特性 闭包的优缺点 闭包的作用 闭包的注意点 💎什么是闭包? 一个函数和对其周围状态的引用捆绑在一起,这样的组合就是闭包. 通俗的说: 一个内层函数可以访问外层函数的作用域 就叫 闭包。 在 JavaScript 中,每当创建一个函数,闭包就会在函数创建的同时被创建出来。 闭包的形成与变量的作用域以及变量的生存周期密切相关。 💎闭包的特性 函数嵌套函数 函数内部可以引用外部的参数和变量 参数和变量不会被垃圾回收机制回收 💎闭包的优缺点
程序员海军
2021/03/25
4290
🌞 深入剖析 JavaScript 闭包
闭包初识
[详情参考](http://www.cnblogs.com/wangfupeng1988/p/3994065.html);
天天_哥
2018/09/29
3120
JavaScript闭包实例讲解
闭包是JavaScript语言中的难点,很多刚入行的(包括我在内)一时对他很难理解,于是在网上各种搜罗有关闭包的学习资料,但是无数的文章介绍闭包,但都是了解一个皮毛。说实在的我到现在也不敢和大家百分百的肯定掌握它。所以今天我就把我的整理的学习笔记分享给大家,希望能够对大家有用。
ZEHAN
2020/09/23
6950
浅谈闭包
闭包 – closure, 应该可以说是javascript的一个难点吧, 其实说难也不难, 只是因为没有真正一个权威的人/书去给他一个真正的定义。 不过,学编程的人一路都要有自己的理解,很少人乐意去v死记一个定义。 自己看闭包也有很久了,但是每次回想起来又忘了自己改如何形容它, 它真的很难吗?其实并没有,每次看别人的博客,还是很快就能理清楚思路的, 于是终究还是要自己写下来,自己给自己一个理解。
用户1394570
2018/08/08
3910
JavaScript入门总结第四弹——函数+十分钟了解闭包
Hello~~偶又来咯,昨天有小可爱说数组有点随意,其实数组的应用的比较广泛,但是并不是很难,主要是不容易都记住,所以兔妞就是给大家将数组进行了一下梳理,总结了一下使用技巧呢~~今天是函数专题呢,也主要为大家揭秘一下闭包~
萌兔IT
2019/07/25
4020
JavaScript入门总结第四弹——函数+十分钟了解闭包
【面试题解】初识 JavaScript 闭包
本文很基础,适合没有了解过闭包的同学入门,我是经常使用 js 书写业务逻辑,但没有刻意使用过闭包,如果你的情况跟我差不多,那么跟着本文,你一定也可以有所收获。
一尾流莺
2022/12/10
2630
【面试题解】初识 JavaScript 闭包
JavaScript——作用域和闭包
本文着重于对 JavaScript 中的作用域和闭包机制进行剖析和说明,闭包本质上也是作用域的一种类型,因为在 JavaScript 里非常重要,所以我们把它在标题里单独列出来。
Html5知典
2019/11/26
7970
JavaScript闭包
IMWeb前端团队
2017/12/29
7950
JavaScript闭包
做前端的可太需要了解闭包了,几乎每个面试都会问到闭包,闭包的重要性不言而喻。什么是闭包:闭包一般是指那些引用了另一个函数作用域中变量的函数,通常是在嵌套函数中实现的。也就是说,闭包让你可以在一个内层函数中访问到其外层函数的作用域。
大熊G
2022/11/14
4210
javascript深入理解js闭包
一、变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域。 变量的作用域无非就是两种:全局变量和局部变量。 Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量。 Js代码 var n=999;   function f1(){     alert(n);   }   f1(); // 999 另一方面,在函数外部自然无法读取函数内的局部变量。 Js代码 function f1(){     var n=999;   }   alert(n); // error
用户1257215
2018/01/30
1.1K0
javascript深入理解js闭包
理解JavaScript闭包
这里有一个地方需要注意,函数内部声明变量的时候,一定要使用var命令。如果不用的话,你实际上声明了一个全局变量!
OECOM
2020/07/01
5850
深入浅出JavaScript之闭包(Closure)
闭包(closure)是掌握Javascript从人门到深入一个非常重要的门槛,它是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现。下面写下我的学习笔记~
哲洛不闹
2018/09/14
4270
深入浅出JavaScript之闭包(Closure)
JavaScript进阶教程(5)-一文让你搞懂作用域链和闭包
在JS中变量可以分为局部变量和全局变量,对于变量不熟悉的可以看一下我这篇文章:搞懂JavaScript全局变量与局部变量,看这篇文章就够了 作用域就是变量的使用范围,分为局部作用域和全局作用域,局部变量的使用范围为局部作用域,全局变量的使用范围是全局作用域。在 ECMAScript 2015 引入let 关键字之前,js中没有块级作用域---即在JS中一对花括号({})中定义的变量,依然可以在花括号外面使用。
AlbertYang
2020/09/16
3710
JavaScript进阶教程(5)-一文让你搞懂作用域链和闭包
JavaScript 闭包
闭包(closure)是 Javascript 语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现。
RiemannHypothesis
2022/10/28
4330
学习Javascript闭包(Closure)
闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现。 下面就是我的学习笔记,对于Javascript初学者应该是很有用的。 一、变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域。 变量的作用域无非就是两种:全局变量和局部变量。 Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量。   var n=999;   function f1(){     alert(n);   }   f1(); // 99
ruanyf
2018/04/12
5760
JavaScript闭包理解
  之前总觉得闭包(Closure)很抽象而且难理解,百度一下"闭包"名词,百度的解释是:“闭包是指可以包含自由(未绑定到特定对象)变量的代码块;这些变量不是在这个代码块内或者任何全局上下文中定义的,而是在定义代码块的环境中定义(局部变量)。看了半天,也没有看懂闭包是什么?以下将是我对闭包(Closure)的理解,如有错误欢迎指出
小蔚
2019/09/11
4050
函数(function)的前世今生
函数就是一段可以反复调用的代码块。函数还能接受输入的参数,不同的参数会返回不同的值。
江米小枣
2020/06/16
7750
函数(function)的前世今生
javascript 闭包详解
一、什么是匿名函数 创建一个函数并将它赋值给变量functionName,这种情况下创建的函数,即匿名函数。(函数表达式就是匿名函数) 二、闭包 1.什么是闭包? 闭包就是能够读取其他函数内部变量的函数。 只有函数内部的子函数才能读取局部变量,因此可以把闭包简单理解成“定义在一个函数内部的函数”。 我们只要把f2作为返回值,我们不就可以在f1外部读取它的内部变量了吗! function f1 () { var num = 1; function f2() { console.
柴小智
2018/04/10
6230
相关推荐
Javascript闭包剖析(通俗易懂)
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验