返回顶部
首页 > 资讯 > 前端开发 > JavaScript >JavaScript高级之闭包详解
  • 150
分享到

JavaScript高级之闭包详解

2024-04-02 19:04:59 150人浏览 泡泡鱼
摘要

目录1. 闭包的概念知识点的补充:2. 闭包的作用:3. 闭包示例3.1 点击li,输出当前li的索引号总结1. 闭包的概念 来看一般函数的执行和启发: func

1. 闭包的概念

来看一般函数的执行和启发:


        function stop() {
            var num = 0;
            console.log(num);
        }
        stop(); // 打印出来num是0
        console.log(num); // 报错 函数未定义

1. 此时,函数的外部无法访问函数内部的变量

2. 函数内部定义的变量不会一直存在,随着函数的运行结束而消失

闭包的概念:

1. 是一个函数,这个函数有权访问另一个作用域中的变量。

2. 另一种说法,当内部函数的生命周期大于外部函数的声明周期,而内部函数以某一种方式被外部作用域访问时,闭包就产生了。

来看如下闭包的代码和解释:


        function fn() {
            var num = 10;
            // function fun() {
            //     console.log(num);
 
            // }
            // return fun;
            return function () {
                console.log(num); // 10
            }
        }
        var f = fn();
        f();

我们可以拆解为几个部分:

1. fn函数里面有内部的返回值且就是一个函数。

2. return的这个函数内部打印了num变量。为什么能够打印num变量,原因在于作用域链的访问机制,下面会补充作用域和作用域链的知识点。

3. 我们在外部用f变量接受了fn(),也就是接受了fn的返回值【内部函数】

4. 紧接着调用f,也就是调用了fn里面的内部函数。最终能够打印10

知识点的补充:

1. 作用域:

变量在某个范围内起作用,超出了这个范围,就不起作用。这个范围就是作用域。作用域在函数的定义时就产生,而不是函数调用时产生的。

2. 作用域链:

一句话概括:根据【内部函数可以访问外部函数变量】,采用就近原则一层一层向上查找变量,这个机制就叫作作用域链。

函数A包含了函数B,那么函数B就是函数A的内部函数,

而内部函数如果要使用一个变量,首先看自己内部有没有这个变量,

如果没有,就会去紧挨着的上一级查找,【就近原则】

如果函数一层一层都找不到,最后才会去全局变量下面找。


        var a = 1;
        var b = 11;
        function fn1() {
            var a = 2;
            var b = '22';
            fn2();
            function fn2() {
                var a = 3;
                fn3();
                function fn3() {
                    var a = 4;
                    console.log(a); // 4
                    console.log(b); // '22'
                }
            }
        }
        fn1();

3. 垃圾回收机制

可以参考这位大哥对于js垃圾回收机制的描述:

//www.jb51.net/article/229425.htm

我们结合这三个概念看闭包的作用

2. 闭包的作用:

我们把函数A叫作外层的函数,这个函数内部有一个函数B。

外部用一个变量f接受函数A的返回值【函数B】

而函数A作用域的变量叫作num

1. 能够在函数的外部访问函数内部的变量【搭建外部访问内部作用域的通道】

原理:上面其实有解释过。

第一要理解,作用链的原理看上面。函数B能够调用函数A的变量num

第二要理解,首先函数A的返回值是函数B【内部函数】,其次这个返回值要在函数外部用变量f接受,接受以后就能够调用函数B,函数B就会访问函数A的变量num。而这个内部函数B就是闭包函数啦。

2. 能够延长函数内部变量的生命周期。

第一个作用带来第二个作用。js的变量存在垃圾回收机制,如果函数执行完毕,变量会被清除,内存也会消除。可是如果利用闭包,变量可以不被立即清除。

原因是,外部的变量f接受了一个函数A的内部函数B,而这个内部函数访问了函数A作用域的变量num,只要函数B执行且变量f一直存在,那么变量num就会一直存在。不会因为函数A的执行结束就消失。

参考了下面的文章,讲的非常详细,推荐看。

javascript闭包详解

3. 闭包示例

后面会补充闭包的一些应用。

我们要想起什么场合用闭包,闭包不能滥用。

3.1 点击li,输出当前li的索引号


    <ul class="nav">
        <li>榴莲</li>
        <li>臭豆腐</li>
        <li>鲱鱼罐头</li>
        <li>大猪蹄子</li>
    </ul>
    <script>
        // 闭包应用-点击li输出当前li的索引号
        // 1. 我们可以利用动态添加属性的方式
        var lis = document.querySelector('.nav').querySelectorAll('li');
        for (var i = 0; i < lis.length; i++) {
            lis[i].onclick = function () {
                console.log(i); // 四个4
            }
        }
    </script>

原理:上图这样写,打印出来的i永远都是4。原因是,此时首先是非严格模式,在非严格模式下,for循环是同步执行任务,而按钮点击再执行是异步任务,同步执行完毕,i加到了4.再执行异步任务打印i,都是4。

改法1:用闭包

1.for循环生成四个立即执行函数

2. 立即执行函数是闭包的一种应用。立即执行函数里面的所有函数包括【点击 回调】函数都可以使用立即执行函数的传递的形参。


        for (var i = 0; i < lis.length; i++) {
            (function (i) {
                // console.log(i);
                lis[i].onclick = function () {
                    console.log(i);
 
                }
            })(i);
        }

改法2:var--->let

点击对应小li,打印i是对应索引号。使用let是es6语法,此时for有块级作用域


        var lis = document.querySelector('.nav').querySelectorAll('li');
        for (let i = 0; i < lis.length; i++) {
            lis[i].onclick = function () {
                // console.log(i);
                console.log(i);
            }
        }

改法3:用设置自定义属性index的方法


        var lis = document.querySelector('.nav').querySelectorAll('li');
        for (var i = 0; i < lis.length; i++) { // 注意这里是var不是let
            lis[i].index = i; // 注意这里是lis[i]不是this.index,此时没有点击,哪里来的this
            lis[i].onclick = function () {
                console.log(this.index);
            }
        }

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注编程网的更多内容!

--结束END--

本文标题: JavaScript高级之闭包详解

本文链接: https://lsjlt.com/news/160205.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • JavaScript高级之闭包详解
    目录1. 闭包的概念知识点的补充:2. 闭包的作用:3. 闭包示例3.1 点击li,输出当前li的索引号总结1. 闭包的概念 来看一般函数的执行和启发: func...
    99+
    2024-04-02
  • python高级语法之闭包和装饰器详解
    目录一、闭包二、一个简单的例子三、装饰器3.1 简单装饰器3.1.1 使用装饰器的语法糖3.1.2 装饰器的执行时机3.2 通用装饰器3.2.1 装饰带有参数的函数3.2.2. 装饰...
    99+
    2024-04-02
  • JavaScript闭包详解
    目录1.什么是闭包?1.2 闭包的记忆性:函数能够记住定义时所处的环境1.3 闭包现象:JS中每次创建函数时都会创建闭包2.闭包的功能:记忆性、模拟私有变量2.1 记忆性2.2 模拟...
    99+
    2024-04-02
  • 详解JavaScript闭包问题
    闭包是纯函数式编程语言的传统特性之一。通过将闭包视为核心语言构件的组成部分,JavaScript语言展示了其与函数式编程语言的紧密联系。由于能够简化复杂的操作,闭包在主流JavaSc...
    99+
    2024-04-02
  • kotlin之闭包案例详解
    闭包,函数式编程福音 先了解函数式编程(Functional Programming) 概念:它属于“结构化编程”的一种,主要思想是把运算过程尽量写成一系列嵌套的函数调用。函数式编程...
    99+
    2024-04-02
  • 一文详解JavaScript中的闭包
    JavaScript 闭包是一种重要的概念,在 JavaScript 编程中被广泛使用。尽管它可能会让初学者感到困惑,但它是理解 JavaScript 语言核心的关键概念之一。本文将深入探讨 JavaScript 闭包,让你了解它是如何工作...
    99+
    2023-05-14
    闭包 前端 JavaScript
  • JavaScript闭包closure详述
    目录一、什么是闭包二、闭包的作用前言: 在JavaScript部分,闭包是很重要的东西,所以我们今天就闭包的相关知识做一总结。首先,在了解闭包前,我们先要知道作用域的相关知识,前面 ...
    99+
    2024-04-02
  • JavaScript 闭包的技巧:掌握高级编程技术
    闭包是JavaScript中强大的技术,它允许函数访问在其创建时存在的外部变量,即使函数执行完毕。掌握闭包的技巧可以显着提高您的代码的灵活性、可维护性和性能。 1. 利用闭包进行数据封装 闭包可以用于封装数据,使其私有化并只对创建它们的函...
    99+
    2024-03-14
    闭包
  • JavaScript闭包原理及作用详解
    目录简介闭包的用途柯里化实现公有变量缓存封装(属性私有化)闭包的原理垃圾收集简介实际开发中的优化简介 说明 本文介绍JavaScript的闭包的作用、用途及其原理。 闭包的定义 闭包...
    99+
    2024-04-02
  • JavaScript总结分享之闭包
    以上就是JavaScript总结分享之闭包的详细内容,更多请关注编程网其它相关文章!...
    99+
    2022-11-22
    前端 JavaScript
  • 闭包之谜:深入探讨 JavaScript 闭包的行为
    范围和作用域链 闭包的作用域链由以下作用域组成: 局部作用域:闭包函数自身的作用域,其中包含闭包中声明的变量。 闭包作用域:创建闭包的函数的作用域,其中包含闭包访问的变量。 全局作用域:浏览器窗口或 Node.js 全局环境。 闭包可...
    99+
    2024-03-14
    闭包
  • 关于Javascript闭包与应用的详解
    目录前言一、什么是闭包?1.1闭包满足条件代码1.2闭包产生时机二、常见闭包示例2.1 子函数做为实参传递2.2 计数器使用(函数返回)三、闭包作用3.1 闭包作用四、闭包应用4.1...
    99+
    2024-04-02
  • 一文详解JavaScript闭包典型应用
    目录1.应用1.1 模拟私有变量1.2 柯里化1.3 偏函数1.4 防抖1.5 节流2.性能问题2.1 内存泄漏2.2 常见的内存泄漏3.闭包与循环体3.1 这段代码输出啥3.2 改...
    99+
    2024-04-02
  • python 闭包详解
    闭包(closure)是函数式编程的重要的语法结构。函数式编程是一种编程范式 (而面向过程编程和面向对象编程也都是编程范式)。在面向过程编程中,我们见到过函数(function);在面向对象编程中,我们见过对象(object)。函数和对象的...
    99+
    2023-01-31
    详解 python
  • JavaScript高级 ES7-ES13 新特性详解
    目录1. ES71. Array Includes2. 指数exponentiation运算符2. ES81. Object values entries2. String Padd...
    99+
    2023-02-06
    js ES7-ES13 新特性 ES7-ES13 新特性
  • 详解JavaScript高级正则表达式
    目录JavaScript高级正则表达式1.正则表达式概述1.1什么是正则表达式1.2正则表达式的特点2.正则表达式在js中的使用2.1正则表达式的创建2.2测试正则表达式3.正则表达...
    99+
    2024-04-02
  • JavaScript内存管理与闭包实例详解
    目录1. 内存管理的理解1.1 认识内存管理1.2 JavaScript的内存管理2. 垃圾回收(GC)2.1 认识垃圾回收2.2 GC算法 – 引用计数2.3 GC算法...
    99+
    2024-04-02
  • PHP高级特性:深入解析闭包的奥秘
    答案: php 闭包是一种匿名函数,可以在定义作用域外访问变量。详细描述:闭包创建:使用 function 关键字创建,可以访问定义作用域内的变量。访问变量:闭包可从内部读取外部变量,访...
    99+
    2024-05-15
    php 闭包 作用域
  • 怎么理解javascript闭包
    本篇内容介绍了“怎么理解javascript闭包”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!第一梯队理解...
    99+
    2024-04-02
  • JavaScript闭包怎么理解
    这篇文章主要介绍“JavaScript闭包怎么理解”,在日常操作中,相信很多人在JavaScript闭包怎么理解问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”JavaScri...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作