闭包及立即执行函数

闭包

  1. 你不需要知道闭包,依然可以把js用得很溜
  2. 把基础搞清楚,闭包自然就理解了

变量的生命周期

  • 默认作用域消失时,内存就被回收(所以变量的生命周期就是作用域的生命周期)
<script>
    var a=1;
</script>

在script全局中声明一个var a=1,当代码执行到这一句时,a=1就出生了(在此之前因为变量声明提升所以a值为undefined,什么都不是),当关闭浏览器页面或刷新页面时,a就死了,刷新后的a是一个新的a,一般来说js变量的生命周期不可能超过页面的生命周期。
所以全局环境下的变量的生命周期就是window窗口的生命周期

<script>
    function f1(){
        var a=1; //也就是在这一行a有值了
        return undefined;
    }
    //浏览器执行到这一行,页面中还不存在a
    f1() //在这一行之后,a就诞生了
    //f1()执行完了之后a就死了
    f1() //再次调用产生的是新的a
    //f1()执行完以后新的a也死掉
</script>

如果a在一个函数里,当你写完这个函数时页面中还不存在a这个变量,只有当执行了f1,且到了var a=1这一行之后,a=1才出生,之前是undedined,那么a什么时候死呢?
当a所在的环境不见了的时候a就死掉了
那么环境什么时候不见呢?
当执行完var a=1这一句之后return,如果没有return浏览器会默认return一个undefined,return执行后回到f1(),开始执行下一行,所在下一行的时候,a就死了(也就是说f1执行完以后a就死了),a所对应的内存就释放了可以给别人用。再调用一次f1时,产生的是一个新的a,执行完函数后新的a也会死。
所以函数中声明的变量一般来说就是函数执行的生命周期,函数开始执行,它就出生了,函数执行完了,它就死了。

  • 如果变量被引用着内存则不能被回收
<script>
    function f1(){
        var a={name:'a'}; //也就是在这一行a有值了,常量a=1是一样的意思,这里用对象来举例
        var b=2;
        window.xxx=a;
        return undefined;
    }
    //浏览器执行到这一行,页面中还不存在a和b
    f1() //在这一行之后,a和b就诞生了
    //f1()执行完了之后b就死了,a为什么没有死,因为a在被别人引用。
    console.log(window.xxx); //{name:'a'},通过这种方法将a暴露出来,可以理解f1执行完之后变量a这个名字死了,但它的值和内存还在,函数执行完以后还可以访问到,那么内存什么时候释放呢?当引用它的window死了,它就死了。
    window.xxx={name:'b'} //如果window.xxx指向了别的东西,a已经没用了,a就可以死了,并不是覆盖,是把新的对象赋值给了xxx,是赋值,赋值,赋值。
</script>

var作用域

  • 就近原则
<script>
    var a;
    function f1(){
        var a; 
        function f2(){
            a=1; //只看父级,a=1对应的是f1里的a
        }
        function f3(){
            var a;
        }
    }
</script>
<script>
    function f2(){}
    function f1(){
        function f2(){}
        f2() //执行的就是上面一行的f2,一层层往外查找。
    }
</script>
  • 词法作用域
    无论代码是否执行,只要看层级关系就能知道给变量a赋的值对应的是哪一个变量,这就是词法作用域。
<script>
    var a;
    function f1(){
        var a; 
        function f2(){
            var a;
            f3(); //执行f3,指的是全局中的a,因为函数最终结果与在哪执行无关,只与初始所在环境有关
            a=1; 
        }
    }
    function f3(){
        a=2;
    }
</script>
  • 同名的不同变量
  • 以上代码中f1()中的a和f2()中的a是不同的两个变量

立即执行函数

当函数不被调用时,函数的内容解析器看都不会看(语法的错误会检查,逻辑错误不会被检查)

  • 想得到一个独立的作用域,那么要声明一个函数
  • 想运行代码,必须执行(调用)声明的函数

假设我们的目的是不要全局变量(函数也是变量),不要全局变量是因为实际工作中一个js代码由多个人编写,全局变量容易和别人的代码发生冲突。

//function f1(){
    //var a;
    //a=1;
    //console.log(a);
//} 注释掉的几行就是函数f1,将函数f1替换到下面,并且把函数名f1去掉
f1();
//改后如下
function(){
    var a;
    a=1;
    console.log(a);
}()
//这样修改的好处是:去掉了全局变量f1,声明了一个匿名函数立即执行
坏处:语法不对,报错
于是很多人放弃了这一写法,直到有一天有人不小心在function加了!,!的作用是对函数运行之后的结果求值,这个值并不重要,因为我们要的是作用域。
//再次修改,声明了一个函数并立即执行,创建了一个独立的作用域,!是为了不报错,除了!,-、+、~都可以,尽量不要括号,容易出问题。

//写法一
!function(){
    var a;
    a=1;
    console.log(a);
}()

//写法二
function f2(){
    var a;
    a=2;
    console.log(a);
}
f2();

写法一和写法二都是为了产生独立的作用域,避免发生代码冲突,我们就可以在定义的函数作用域内为所欲为地声明任何变量。
全局变量是邪恶的,永远不要用。
怎样避免写全局变量呢?
在ES6之前只有一种方法,就是利用函数里的局部变量,这就是立即执行函数的意义所在。

//写法1
function sss(p1,p2){}
//写法2
function sss(){
    var p1=arguments[0];//arguments[0]就是外面传进来的第一个参数
    var p2=arguments[1];//arguments[1]就是外面传进来的第二个参数
}
//写法一和写法二基本等价,都是在函数作用域内声明的p1和p2

//所以以下两种写法也是等价的
//写法三
!function(a){ 
    a=1;
    console.log(a);
}()

//写法四
!function(){
    var a=arguments[0]; //arguments就是外面传进来的参数(也就是函数花括号后面的小括号中传进来的参数,传进来啥呢?啥也没有)
    a=1;
    console.log(a);
}(/*没有传参*/)
//所以arguments是一个空的数组,所以先声明一个变量a初始化为undefined,接着给a赋值1
//如果在全局中声明一个a并赋值为100,改写代码:
var a=100;
!function(a){
    a=1; ️
    console.log(a); //
}()
console.log(a);//如果传递给function函数的参数a指的是全局中的a,那么全局中打印a得到的应该为1;如果传递给function函数的参数a指的是function形参中声明的a,那么全局中打印a得到的应该还是100,那么实际结果究竟是多少呢?
//结果:里面的a打印为1,外面的a打印为100
//结论:传递给function()的参数a是形参声明的变量,值就是传进去的第一个参数。

//什么是传进去的第一个参数呢?
var a=100
!function(a){
    console.log(a); //99,相当于var a=第一个传递进来的参数
}(99)
console.log(a); //100

//如果有多个参数呢?
var a=100;
!function(a,b){
    console.log(a,b); //99 "hello",相当于var a=第一个传递进来的参数arguments[0],var b=第二个传递进来的参数arguments[1]
}(99,'hello')
console.log(a);

//结论:传递给function的参数a,b是全新的a和b,属于函数的作用域内的变量,与外面无关。

//最关键的来了!
var a=100;
!function(a){//第一行的a
    console.log(a); 
}(a)//第二行的a
console.log(a);

//那么第一行的a和第二行的a是同一个a吗?
//当然不是!
//第一行的a是函数作用域内的a,第二行的a是全局作用域中的a,以上代码中,只是恰好在执行函数时把全局中的a赋值给函数中声明的a,这也印证了前面所说的,不同作用域中的同名变量是不同的。

//例子来印证
//例1
var a=100;
!function(a){
    console.log(a); //100,因为函数内只是定义了a但并没有给a赋值,所以会去找外层作用域中有没有变量a,找到了就返回外层作用域中a的值
}(a)
console.log(a); //100,全局作用域下a为100

//例2
var a=100;
!function(a){
var a=1;
    console.log(a); //1,函数的作用域中定义了a为1,所以打印a为1
}(a)
console.log(a); //100,全局中a为100

总结一下立即执行函数:

var a = 1;
function xxx(a){
    a = 2;
}
xxx(a);
console.log(a);

结论:xxx是废话,删掉,删掉后就是匿名函数,接着语法报错,加个运算符,改写函数就成了立即执行函数。

var a = 1
!function (a) {
    a = 2
    console.log(a)//2
}(a)
console.log(a)//1

变量(声明)提升

  • 浏览器在执行代码之前,会先把所有声明提升到作用域的顶部
  • 你却从来都不知道去提升一下变量
  • 只要你提升变量,面试题就是小case
function a(){};
var a=100;
console.log(a); //100
var a=100;
function a(){};
console.log(a); //100
console.log(a); //function a(){}
var a=100;
function a(){};
console.log(a);//undefined
var a=100;
console.log(a);//100
var a=100;
var a=function(){};
function a(){};
console.log(a);//function(){}

//提升改写以下更直观,一定要提升了再看代码,不然一定错,提升只会发生在当前作用域内
var a;
var a;
function a(){};
a=100;
a=function(){};
console.log(a);//function(){} 

然后很多时候想要提升却无从下手

var a=100;//(1)
f1();
function f1(){
    var b=2;
    if(b===1){
        var a;//(2)
    }
    a=99; //这里的a指的是第一个a还是第二个a?
}

//提升一下先
var a;
function f1(){
    var b;
    var a;//这里从if里直接提升到外面是因为js里没有块级作用域,只有函数作用域和全局的概念(后面会学到的let有块级作用域),所以if、while、for等流控制语句中的变量都会提升到包含它的函数或者全局中
    b=2;
    if(b===1){
    }
    a=99; //结论:当然是第二个a
}
a=100;
f1();

时机(异步)

<body>
<button id="button">click me</button>
<script>
var button=document.querySelector('button');
button.onclick=function f(){
    console.log('A');//只有用户点击了button的情况下A才会被打印出来
}
console.log('B');//正常情况下B一定会被打印出来
//所以B一定在A前面被打印出来

//当用户点击按钮时,浏览器会执行函数f,把函数f挂载到onclick上,即button.onclick,onclick的执行是不确定的,所以当浏览器执行它时会手动执行button.onclick.call(target,event)这一句代码,这就是异步,先写的代码后执行,后写的代码先执行。
</script>
</body>

面试题

<body>
    <ul>
        <li>选项1</li>
        <li>选项2</li>
        <li>选项3</li>
        <li>选项4</li>
        <li>选项5</li>
        <li>选项6</li>
    </ul>
    <script>
        var items = document.querySelectorAll('li');
        for (var i = 0; i < items.length; i++) {
            items[i].onclick = function () {
                console.log(i);
            }
        }
    </script>
</body>

改写一下代码~

<script>
    //获取页面上的所有li,items是一个伪数组
    var items;
    var i;
    items = document.querySelectorAll('li');
    for (i = 0; i < items.length; i++) {
        //i==0,1,2,3,4,5
        items[i].onclick = function () {
            console.log(i);//C
        }
    }
    //i==6
    console.log(i);//D
    //D处的i肯定会被打印,C处的i不一定被打印,只有用户点击时才会有打印值。不管有没有D处的代码,C处的代码都是后执行。
    //D处的i就是全局中的i,也就是for循环里提升出去的i,for循环开始执行后,过程为i=0,item[0].onclick=console.log(i),此时i还是一个变量,值是不确定的;i=1,item[1].onclick=console.log(i),此时i还是一个变量,值是不确定的;……依此类推到i=5,item[5].onclick=console.log(i),执行完循环之后,i=5,i++,D处打印结果为6,此时用户点击li元素,控制台打印C处的i,也就是全局中的i,值为6。
    //因为C处的肯定在D之后执行,而他们打印的都是全局中的i,所以点击的每一个li打印出的都是6
</script>

可是如果这不是你想要的结果,你希望的是点击页面上的6个li得到的是0,1,2,3,4,5而不是6个6,那么怎样做呢?
试着改写一下函数:

for (i = 0; i < items.length; i++) {
            //6次循环中,函数被创建了6次,这不是一个声明,而是一个赋值,6个函数就有6个j,每一个j都等于不同的值。i只有一个。
            var temp = function (j) {
                console.log(j);//0,1,2,3,4,5
                items[j].onclick = function () {
                    console.log(j);//用户点击对应的也是0,1,2,3,4,5
                }
            }
            temp(i);
        }

精髓是:以前是一个变量i贯穿整个6次循环,现在是每一次循环都是一个新的变量j,那么怎样得到一个新的变量呢?直接在for循环里声明var j肯定不行,因为变量j会被提升到全局作用域中,与i的效果相同,所以必须创建一个函数,这个函数就是一个新的作用域,就会有6个不同的j。

以上代码中出现了一个前面的知识点————立即执行函数,所以还可以改写一下

for (i = 0; i < items.length; i++) {
    !function (j) {
        items[j].onclick = function () {
            console.log(j);
        }
    }(i);
}

还有没有别的方法呢?有

for (i = 0; i < items.length; i++) {
    function temp(j) {
        return function () {
            console.log(j);
        }
    }
    var fn = temp(i);
    items[i].onclick = fn;
}

优化上述代码,j写为i(改写后的i与全局中的i不是同一个,只是同名而已),转成立即执行函数

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

去掉变量fn,然后得到的就是回答这个面试题的标准答案

for (i = 0; i < items.length; i++) {
    items[i].onclick = function (i) {
        return function () {
            console.log(i);
        }
    }(i);
}

解释:因为用了闭包,所以出现了问题(闭包造成的),但这不是闭包的问题,是声明会提升的问题,所以声明一个新的作用域阻隔变量提升问题,解决办法是使用立即执行函数产生新的作用域。


下面再上一个相似的面试题目

var arr = [];
    for (var i = 0; i < 6; i++) {
        arr[i] = function () {
            console.log(i);//6,因为i是贯穿全局的变量
    }
}
console.log(arr[3]());

改写一下

var arr = [];
    for (var i = 0; i < 6; i++) {
        arr[i] = function () {
            console.log(i);//这里没有异步,但fn执行前i已经是6了,所以打印6,因为i是贯穿全局的变量
    }
}
let fn=arr[3];
i===6;
fn();//去执行console.log(i),得到6

如果想要console.log(i)得到的是0,1,2……这种形式的,只需要创建一个作用域:

var fnArr = [];
for (var i = 0; i < 10; i++) {
    var temp = function (i) {
        fnArr[i] = function () {
            return i;
        }
    }
    temp(i);
}
console.log(fnArr[3]()); //3

函数return函数也可

var fnArr = [];
for (var i = 0; i < 10; i++) {
    fnArr[i] = function (i) {
        return function () {
            return i;
        }
    }(i);
}
console.log(fnArr[3]());

上面的代码再优化一下,改成立即执行函数

var fnArr = [];
for (var i = 0; i < 10; i++) {
    !function (i) {
        fnArr[i] = function () {
            return i;
        }
    }(i);
}
console.log(fnArr[3]());

最简单的,通过let创建块级作用域

var fnArr = [];
for (let i = 0; i < 10; i++) {
    fnArr[i] = function () {
        return i;
    };
}
console.log(fnArr[3]());

闭包

声明一个变量,在函数里去使用这个变量,外面声明的变量加上整个函数体就是一个闭包。如下所示

var local='变量';
function foo(){
    console.log(local)
}

知道什么是闭包没有任何意义,但和垃圾回收结合在一起就有意义了,

var fn=function(){
    //b不会被引用,但b占用的内存也不会被回收,这就是内存泄漏,但不是闭包的错,是IE的锅
    var b={
        name:'b'
    }
    //以下就是闭包
    var a={
        name:'a'
    }
    
    return function(){
        return a
    }
}()
//
console.log(fn())
//谁也不能访问b了

面试官问闭包,其实就是问的立即执行函数,问题是闭包造成的,解决办法是立即执行函数,原理是创建新的作用域。
闭包有什么作用:暴露局部变量,虽然外面访问不到,但可以通过函数去间接地操作它。
立即执行函数专克闭包。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容