浅谈自执行函数(立即调用的函数表达式)

在JavaScript中,会遇到自执行匿名函数:(function () {/*code*/} ) ()
这个结构大家并不陌生,但若要说:为什么要括弧起来?它的应用场景有哪些?……就会有点模糊。
此处作个小结。

本文篇幅比较长,但例子都很简单,可以跳跃式阅读。

一、函数的声明与执行

我们先来看下最初的函数声明与执行:

    // 声明函数fun0
    function fun0(){
        console.log("fun0");
    }

    //执行函数fun0
    fun0(); // fun0

除了上面这种最常见的函数声明方式,还有变量赋值方式的,如下:

    // 声明函数fun1 - 变量方式
    var fun1 = function(){
        console.log("fun1");
    }

    // 执行函数fun1
    fun1(); // fun1

二、 函数的一点猜想

既然函数名加上括号fun1()就是执行函数。
思考:直接取赋值符号右侧的内容直接加个括号,是否也能执行?
试验如下,直接加上小括弧:

    function(){
        console.log("fun");
    }();

以上会报错 line1:Uncaught SyntaxError: Unexpected token (
分析: function 是声明函数关键字,若非变量赋值方式声明函数,默认其后面需要跟上函数名的。

加上函数名看看:

    function fun2(){
        console.log("fun2");
    }();

以上会报错 line3:Uncaught SyntaxError: Unexpected token )
分析: 声明函数的结构花括弧后面不能有其他符号(比如此处的小括弧)。

不死心的再胡乱试一下,给它加个实参(表达式):

    function fun3(){
        console.log("fun3");
    }(1);

不会报错,但不会输出结果fun3
分析: 以上代码相当于在声明函数后,又声明了一个毫无关系的表达式。相当于如下代码形式:

    function fun3(){
        console.log("fun3");
    }

(1);

// 若此处执行fun3函数,可以输出结果
fun3(); //"fun3"

三、自执行函数表达式

1. 正儿八经的自执行函数

想要解决上面问题,可以采用小括弧将要执行的代码包含住(方式一),如下:

// 方式一
    (function fun4(){
        console.log("fun4");
    }()); // "fun4"

分析:因为在JavaScript语言中,()里面不能包含语句(只能是表达式),所以解析器在解析到function关键字的时候,会把它们当作function表达式,而不是正常的函数声明。
除了上面直接整个包含住,也可以只包含住函数体(方式二),如下:

// 方式二
    (function fun5(){
        console.log("fun5");
    })();// "fun4"

写法上建议采用方式一(这是参考文的建议。但实际上,我个人觉得方式二比较常见)。

2. “歪瓜裂枣”的自执行函数

除了上面()小括弧可以把function关键字作为函数声明的含义转换成函数表达式外,JavaScript的&& 与操作、||或操作、,逗号等操作符也有这个效果。

    true && function () { console.log("true &&") } (); // "true &&"
    false || function () { console.log("true ||") } (); // "true ||"
    0, function () { console.log("0,") } (); // "0,"

// 此处要注意: &&, || 的短路效应。即: false && (表达式1)  是不会触发表达式1;
// 同理,true || (表达式2) 不会触发表达式2

如果不在意返回值,也不在意代码的可读性,我们甚至还可以使用一元操作符(! ~ - + ),函数同样也会立即执行。

    !function () { console.log("!"); } (); //"!"
    ~function () { console.log("~"); } (); //"~"
    -function () { console.log("-"); } (); //"-"
    +function () { console.log("+"); } (); //"+"

甚至还可以使用new关键字:

// 注意:采用new方式,可以不要再解释花括弧 `}` 后面加小括弧 `()` 
new function () { console.log("new"); } //"new"

// 如果需要传递参数
new function (a) { console.log(a); } ("newwwwwwww"); //"newwwwwwww"

嗯,最好玩的是赋值符号=同样也有此效用(例子中的i变量方式):

//此处 要注意区分 i 和 j 不同之处。前者是函数自执行后返回值给 i ;后者是声明一个函数,函数名为 j 。
    var i = function () { console.log("output i:"); return 10; } (); // "output i:"
    var j = function () { console.log("output j:"); return 99;}
    console.log(i); // 10
    console.log(j); // ƒ () { console.log("output j:"); return 99;}

上面提及到,要注意区分 var ivar j 不同之处(前者是函数自执行后返回值给i ;后者是声明一个函数,函数名为j)。如果是看代码,我们需要查看代码结尾是否有没有()才能区分。一般为了方便开发人员阅读,我们会采用下面这种方式:

    var i2 = (function () { console.log("output i2:"); return 10; } ()); // "output i2:"
    var i3 = (function () { console.log("output i3:"); return 10; }) (); // "output i3:"
// 以上两种都可以,但依旧建议采用第一种 i2 的方式。(个人依旧喜欢第二种i3方式)

四、自执行函数的应用

1. for循环 + setTimeout 例子

直接来看一个例子。for 循环里面通过延时器输出索引 i

for( var i=0;i<3;i++){
    setTimeout(function(){
        console.log(i);
    }
    ,300);
}
// 输出结果 3,3,3

输出结果并不是我们所预想的1,2,3。当然,这个要涉及到setTimeout 的原理了,即使把300ms改成0ms,同样也会输出3,3,3。具体可以查看博文 setTimeout(0) 的作用 。这里摘取其中一段说明。

JavaScript是单线程执行的,无法同时执行多段代码。当某段代码正在执行时,后续任务都必须等待,形成一个队列。只有当前任务执行完毕,才会从队列中取出下一个任务——也就是常说的“阻塞式执行”。

上面代码中设定了一个setTimeout,那浏览器会在合适时间(此处是300ms后)把代码插入任务队列,等待当前的for循环代码执行完毕再执行。(注意:setTimeout 虽然指定了延时的时间,但并不能保证执行的时间与设定的延时时间一直,是否准确取决于 JavaScript 线程是拥挤还是空闲。)

上面说了那么多,都是在分析为什么会输出3,3,3。那怎么样才能输出1,2,3呢?
看看下面的方式(写法一):把setTimeout代码包含在匿名自执行函数里面,就可以实现“锁住”索引i,正常输出索引值。

for( var i=0;i<3;i++){
    (function(lockedIndex){
        setTimeout(function(){
            console.log(lockedIndex);
        }
        ,300);
    })(i);
}
// 输出 "1,2,3"

分析:尽管循环执行结束,i值已经变成了3。但因遇到了自执行函数,当时的i值已经被 lockedIndex锁住了。也可以理解为 自执行函数属于for循环一部分,每次遍历i,自执行函数也会立即执行。所以尽管有延时器,但依旧会保留住立即执行时的i值。
上面的分析有点模糊和牵强,也可以从 闭包 角度出发分析的。但鄙人“闭包”概念模糊,先遗憾下,以后再补充分析了。QAQ

除了上面的写法,也可以直接在 setTimeout 第一个参数做自执行(写法二),如下。
注意: 写法二 会比 写法一 先执行。原因不明。

for( var i=0;i<3;i++){
    setTimeout((function(lockedInIndex){
        console.log(lockedInIndex);
    })(i)
    ,300);
}

关于 自执行函数参数 lockedInIndex ,补充说明以下几点。
注意:自执行函数在 setTimeout 和在 setTimeout 里在第2、3中情况有区别(原因不明,后续再补)。

// 1. lockedInIndex变量,也可以换成i,因为和外面的i不在一个作用域
for( var i=0;i<3;i++){
    (function(i){
        setTimeout(function(){
            console.log(i); // 1,2,3
        }
        ,300);
    })(i);
}

for( var i=0;i<3;i++){
    setTimeout((function(i){
        console.log(i); // 1,2,3
    })(i)
    ,300);
}

// 2. 自执行函数不带入参数 
for( var i=0;i<3;i++){
    (function(){
        setTimeout(function(){
            console.log(i); // 3,3,3
        }
        ,300);
    })();
}

for( var i=0;i<3;i++){
    setTimeout((function(){
        console.log(i); // 1,2,3
    })()
    ,300);
}

// 3. 自执行函数只有实参没有写形参
for( var i=0;i<3;i++){
    (function(){
        setTimeout(function(){
            console.log(i); // 3,3,3
        }
        ,300);
    })(i);
}

for( var i=0;i<3;i++){
    setTimeout((function(){
        console.log(i); // 1,2,3
    })(i)
    ,300);
}

// 4. 自执行函数只有形参没有写实参,这种情况不行。因为会导致输出 undefined。
for( var i=0;i<3;i++){
    (function(i){
        setTimeout(function(){
            console.log(i); // undefined,undefined,undefined
        }
        ,300);
    })();
}

for( var i=0;i<3;i++){
    setTimeout((function(i){
        console.log(i); // undefined,undefined,undefined
    })()
    ,300);
}
2. html元素绑定事件

假设要对页面上的元素安装点击相同的点击事件。我们会考虑如下方式。

<div id="demo">
    <p>p1</p>
    <p>p2</p>
    <p>p3</p>
    <p>p4</p>
    <p>p5</p>
</div>
<script type="text/javascript">
    var oDiv = document.getElementById("demo");
    var eles = oDiv.getElementsByTagName("p");

    for ( var k=0; k < eles.length; k++){
        eles[k].addEventListener('click',function(e){
            alert("index is: " + k + ", and this ele is: " + eles[k]); // index is: 5, and this ele is:undefined
        });

        /** 安装事件方式也可以用 onclick 方式。不过这种方式安装多个onclick触发事件时,只执行最后安装的那一个。 */
        // eles[k].onclick = function(){
        //  alert("index is: " + k + ", and this ele is: " + eles[k]);
        // }
    }
</script>

我们期望点击某个 p元素,能得到该元素所在的索引,但实际是,点击每个p,索引值都是5,而对应的元素都是undefined
分析:这种现象和上面的延时器类似,JavaScript在执行for循环语句时,负责给元素安装点击事件,但当用户点击元素触发事件时,for循环语句早就执行完毕了,此时的 i 自然是5了。

一样的,我们也希望“锁住”索引i。所以可以如上采用自执行函数方式( 在addEventListener外部 ):

/** 1. 自执行函数方式一 */
for ( var k=0; k < eles.length; k++){
        (function(k){
            eles[k].addEventListener('click',function(e){
                alert("index is: " + k + ", and this ele is: " + eles[k].innerHTML); 
            });
        })(k);
    }

也可以 在addEventListener里面 的处理函数使用自执行函数表达式,具体如下。不过上面的方式更具有可读性。

        /** 2. 自执行函数方式二 */
    for ( var k=0; k < eles.length; k++){
            eles[k].addEventListener('click',function(k){
                return function(e){
                    alert("index is: " + k + ", and this ele is: " + eles[k].innerHTML);
                }
            }(k));
    }

当然,除了自执行函数表达式,我们还有一种讨巧的解决办法:

    /** 3. 讨巧的解决方案 */
    for ( var k=0; k < eles.length; k++){
        eles[k].index = k;
        eles[k].addEventListener('click',function(e){
            alert("index is: " + this.index + ", and this ele is: " + eles[this.index].innerHTML);
        });
    }
// 把索引 k 保存在元素的属性中。在点击元素触发事件时,巧用 this 关键字去取出当前点击对象的属性 index,也就是对应的索引。

四、自执行与立即执行

最后来唠嗑下命名方式。
文中对 (function () {/*code*/} ) () 这种表达式,称作为 自执行匿名函数(Self-executing anonymous function);而参考的英文博文中作者更建议称它为 立即调用的函数表达式(Immediately-Invoked Function Expression)。
以下是截取该参考博文的例子:

// 自执行函数。自己调用自己(递归)
function foo() { foo(); }

// 自执行的匿名函数。
var foo = function () { arguments.callee(); }; 

// 立即执行匿名函数。但我们习惯称其为:自执行的匿名函数。
(function () { /* code */ } ());

// 立即执行函数。加一个标示名称,可以方便Debug
(function foo() { /* code */ } ());

// 立即调用的函数表达式(IIFE)也可以自执行,不过可能不常用罢了
(function () { arguments.callee(); } ());
(function foo() { foo(); } ());

注意:arguments.callee在ECMAScript 5 strict mode里被废弃了。

个人愚见:上面例子中把 自执行 解释成 “自己调用自己”,当然和 立即执行 相差很大了。但如果把 自执行 解释成 “自动执行”,就和 立即执行 异曲同工了。
命名方式绝对统一也没必要,重要的是能深入了解并应用它们。

参考内容:

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

推荐阅读更多精彩内容