• JavaScript 再谈闭包


    之前有整理过一版关于闭包的概念,但感觉思路不是很清晰,是临时想起一些例子来讲的,今天再次来讲一下闭包。

    闭包:

    函数嵌套函数,内部函数可以引用外部函数的参数和变量

    function aaa(a){
        var b=5;
        function bbb(){
            alert(a);
            alert(b);
        }
    }    

     在上面的例子当中,bbb函数是可以访问到aaa函数中的a和b的,同时,JS中的垃圾回收机制也不会回收a,b。

    function aaa(){
        var a =5;
        function bbb(){
            alert(a);
        }
        return bbb;
    }
    
    var c = aaa();
    c();

    可以看到此时调用c()是可以弹出5的,即a在调用aaa()之后是没有被回收的。即常驻内存。

    闭包的好处:

    1 希望一个变量可以长期驻扎在内存,之前已经说明。

    2 避免全局变量污染

    var a=1;
    function aaa(){
        a++;
        alert(a);
    }
    aaa();    //2
    aaa();    //3

    此时函数aaa访问的是全局变量a,那么就很容易被其他函数或程序修改。那么可以使用一下代码来避免这个问题

    function aaa(){
        var a=5;
        return function(){
            a++;
            alert(a);
        }
    }
    var b=aaa();
    b();    //6
    b();    //7
    alert(a)  //error

    以上的例子中可以看出已经把a放进函数里作为一个局部变量被引用,a也会常驻内存,不会被垃圾回收清理。

    还可以将其改写成函数声明表达式:将function用括号包围起来做到即时调用,并且减少全局变量的污染,实现代码模块化(即该代码不会因外界条件而改变结果)

    var b=(function() {
        var a=5;
        return function() {
            a++;
            alert(a);
        }
    })();
    b();    //6
    b();    //7
    alert(a)  //error

    3 私有成员

    var aaa=(function(){
        var a =1;
        function bbb(){alert(++a);}
        function ccc(){alert(++a);}
        return{
            b:bbb,
            c:ccc
        }
    
    })() ;
    alert(aaa.b);    //2
    alert(aaa.c);    //3
    alert(a);          //error
    alert(bbb);      //error
    alert(ccc);       //error

    aaa内的函数和变量只能通过aaa来访问,外部是访问不到的,由此实现了私有成员的创建。

    4 循环 索引 作用域延伸

    //有一个3个<li>标签的页面,需要绑定点击事件,弹出对应的序号。
    
    windows.onload=function(){
        var aLi=document.getElementByTagName('li');
        for(var i=0;i<aLi.length;i++){
            aLi[i].onclick = (function(i){
                return function(){alert(i);}
            })(i);
        }
    }

    当然在ES6中使用let替代var也可以解决这个问题。所以在支持ES6的浏览器中能用let的地方就不要用var。

    还有一个需要注意的地方:

    IE下回引发内存泄漏

    //假设有个id为div1的div
    window.onload=function(){
        var oDiv=document.getElementById('div1');
        oDIv.onclick=function(){
            alert(oDiv.id);
        };
    }

    在上述情况中,DOM树中的元素被更深层级的调用,会导致关闭页面后无法释放内存的问题,最终会导致内存泄漏。

    要想解决这个问题也是十分的简单,只需要在关闭界面的时候强制解除对元素的引用。即:

    //假设有个id为div1的div
    window.onload=function(){
        var oDiv=document.getElementById('div1');
        oDIv.onclick=function(){
            alert(oDiv.id);
        };
    //以下为添加解除程序
        window.onunload=function(){
            oDiv.click=null;//或者oDiv=null也可以。
        }
    }

    以上便是闭包的所有相关知识。总结一下:

    1 闭包的含义:函数中嵌套函数,嵌套的内部函数可以访问外部函数的参数和变量。

    2 闭包的作用:

      1 避免全局变量的污染。

      2 创建私有成员(函数和变量)。实现代码模块化

      3 作用域的延伸,也是变量常驻内存的一种体现。

     
  • 相关阅读:
    Python课程回顾(day26)网络编程
    Python课程回顾(day27)
    Python课程回顾(day25)
    Python课程回顾(day24)
    Python课程回顾(day23)
    Python课程回顾(day22)
    Python课程回顾(day21)
    Python课程回顾(day20)
    Python课程回顾(day19)
    hive小tips(各种解析)
  • 原文地址:https://www.cnblogs.com/BigJ/p/8065267.html
Copyright © 2020-2023  润新知