• 技术文章 >web前端 >js教程

    js闭包是什么?对js闭包的理解(附代码)

    不言不言2021-04-14 10:59:09原创42141

    js中闭包closure,是指函数变量可以保存在函数作用域内,因此看起来是函数将变量“包裹”了起来,根据定义,包含变量的函数就是闭包。

    本教程操作环境:windows7系统、javascript1.8.5版、Dell G3电脑。

    最初的定义

    闭包(closure),是指函数变量可以保存在函数作用域内,因此看起来是函数将变量“包裹”了起来。
    //根据定义,包含变量的函数就是闭包
    function foo() {
        var a = 0;
    }
    cosole.log(a) 
    // Uncaught ReferenceError: a is not defined

    《JavaScript高级程序设计》对闭包定义

    闭包是指有权访问另一个函数作用域中的变量的函数

     //根据《JavaScript高级程序设计》,访问上层函数的作用域的内层函数就是闭包
    function foo() {
        var a = 2;
        function bar() {
            console.log(a);
        }
        bar();
    }
    foo();

    《JavaScript权威指南》对闭包定义

    函数对象可以通过作用域链相互关联起来,函数体内部变量可以保存在函数作用域内,这就是闭包。

     var global = "global scope"; //全局变量
    function checkscope() {
        var scope = "local scope"; //局部变量
        function f() {
            return scope; //在作用域中返回这个值
        };
        return f();
    }
    checkscope(); // 返回 "local scope"

    严格来说,闭包需要满足三个条件:

    【1】访问所在作用域;

    【2】函数嵌套;

    【3】在所在作用域外被调用
    有些人觉得只满足条件1就可以,所以IIFE是闭包;有些人觉得满足条件1和2才可以,所以被嵌套的函数才是闭包;有些人觉得3个条件都满足才可以,所以在作用域以外的地方被调用的函数才是闭包

    为什么我们需要闭包,js闭包有什么用处

    首先来看一个例子,我们来实现一个计数器。

    var counter = 0;
    function add() {
       return counter += 1;
    }
    add();
    add();
    add();// 计数器现在为 3

    现在我们已经达到了目的,可是问题来了,代码中的任何一个函数都可以随意改变counter的值,所以这个计数器并不完美。那我们把counter放在add函数里面不就好了么?

    function add() {
        var counter = 0;
        return counter += 1;
    } 
    add();
    add();
    add();// 本意是想输出 3, 但输出的都是 1

    所以这样做的话,每次调用add函数,counter的值都要被初始化为0,还是达不到我们的目的。

    如何使用闭包

    所以这时候我们就要用闭包去解决这个问题了,先看代码。

    var add = (function () {
        var counter = 0;
        return function () {return counter += 1;}
    })();
    add();
    add();
    add();// 计数器为 3

    这时候我们完美实现了计数器。这段非常精简,可以拆分成如下等价代码。

    function outerFunction () {
         var counter = 0;
         function innerFunction (){
             return counter += 1;
         }
         return innerFunction;
    }
    var add = outerFunction();
    add();
    add();
    add();// 计数器为 3

    这时候的add就形成了一个闭包。一个闭包由两部分组成,函数和创建该函数的环境。环境是由环境中的局部变量组成的。对于闭包add来说,它由函数innerFunction和变量counter组成,所以这时候add是可以访问变量counter的。

    使用闭包应注意的问题

    由于闭包会携带包含它的函数的作用域,因此会比其他函数占用更多的内存。因此可以手动解除对匿名函数的引用,以便释放内存。

    function f2(){
        var n=22;
        var nAdd=function(){n++};
        return function(){
            return {
                n:n,
                nAdd:nAdd
            }
        }
    }
    //result2就是创建了一个匿名函数
    var result2=f2();
    //调用函数
    console.log(result2());
    result2().nAdd();
    console.log(result2());
    //解除对匿名函数的引用,以便释放内存
    result2=null;

    相关推荐:

    JS闭包的使用

    简单理解JS闭包

    javascript深入理解js闭包

    以上就是js闭包是什么?对js闭包的理解(附代码)的详细内容,更多请关注php中文网其它相关文章!

    声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn核实处理。
    专题推荐:js闭包
    上一篇:js数组去重的方法有哪些?js数组去重五种方法总结(附代码) 下一篇:自己动手写 PHP MVC 框架(40节精讲/巨细/新人进阶必看)

    相关文章推荐

    • 带你进一步理解js闭包(详细)• js闭包的代码示例讲解• js闭包与作用域链是什么意思?js闭包与作用域链详解• js闭包是什么
    1/1

    PHP中文网