zoukankan      html  css  js  c++  java
  • 详解立即执行函数(function(){}()),(function(){})()

      要知道这几种写法之间的区别,我们要先聊些题外话——js中函数的两种命名方式,即表达式和声明式。

      函数的声明式写法为:function foo(){/*...*/},这种写法会导致函数提升,所有function关键字都会被解释器优先编译,不管是声明在什么位置,都可以调用它,但是它本身不会被执行,定义只是让解释器知道其存在,只有在被调用的时候才会执行。

    图1 声明式函数

      函数的表达式写法为:var foo=function(){/*...*/},这种写法不会导致函数提升,于是就必须先声明,再调用,否则会出错,如图2。

    图2 表达式函数

      现在,回到正题,(function(){}()),(function(){})()这两种是js中立即执行函数的写法,函数表达式后加上()可以被直接调用,但是把整个声明式函数用()包起来的话,则会被编译器认为是函数表达式,从而可以用()来直接调用,如(function foo(){/*...*/})(),但是如果这个括号加在声明式函数后面,如function foo(){/*...*/}(),则会报错,很多博客说这种写法()会被省略,但实际是会出错,因为不符合js的语法,所以想要通过浏览器的语法检查,就必须加点符号,比如()、+、!等,具体可以查看图3。

    图3 立即执行函数

     总结一下就是:

    function foo(){console.log("Hello World!")}()//声明函数后加()会报错
    (function foo(){console.log("Hello World!")}())//用括号把整个表达式包起来,正常执行
    (function foo(){console.log("Hello World!")})()//用括号把函数包起来,正常执行
    !function foo(){console.log("Hello World!")}()//使用!,求反,这里只想通过语法检查。
    +function foo(){console.log("Hello World!")}()//使用+,正常执行
    -function foo(){console.log("Hello World!")}()//使用-,正常执行
    ~function foo(){console.log("Hello World!")}()//使用~,正常执行
    void function foo(){console.log("Hello World!")}()//使用void,正常执行
    new function foo(){console.log("Hello World!")}()//使用new,正常执行

      立即执行函数一般也写成匿名函数,匿名函数写法为function(){/*...*/},就是使用function关键字声明一个函数,但未给函数命名,倘若需要传值,直接将参数写到括号内即可如图4所示。

    图4 立即执行函数的传参

      将它赋予一个变量则创建函数表达式,赋予一个事件则成为事件处理程序等。但是需要注意的是匿名函数不能单独使用,否则会js语法报错,至少要用()包裹起来。上面的例子可以写成如下形式:

    (function(){console.log("我是匿名函数。")}())
    (function(){console.log("我是匿名函数。")})()
    !function(){console.log("我是匿名函数。")}()
    +function(){console.log("我是匿名函数。")}()
    -function(){console.log("我是匿名函数。")}()
    ~function(){console.log("我是匿名函数。")}()
    void function(){console.log("我是匿名函数。")}()
    new function(){console.log("我是匿名函数。")}()
    

      立即执行函数的作用是:1.创建一个独立的作用域,这个作用域里面的变量,外面访问不到,这样就可以避免变量污染。2.闭包和私有数据。提到闭包,不得不提下那道经典的闭包问题。

     1 <ul id=”test”>
     2     <li>这是第一条</li>
     3     <li>这是第二条</li>
     4     <li>这是第三条</li>
     5 </ul>
     6 
     7 <script>
     8     var liList=document.getElementsByTagName('li');
     9     for(var i=0;i<liList.length;i++)
    10     {
    11         liList[i].onclick=function(){
    12             console.log(i);
    13         }
    14     };
    15 </script>

      很多人觉得这样的执行效果是点击第一个li,则会输出1,点击第二个li,则会输出二,以此类推。但是真正的执行效果是,不管点击第几个li,都会输出3,如图5所示。因为 i 是贯穿整个作用域的,而不是给每个 li 分配了一个 i,用户触发的onclick事件之前,for循环已经执行结束了,而for循环执行完的时候i=3。

    图5 各自点击第1,2,3个li,或是之后再次点了多少次,都会输出3,可见,右边控制台输出了8次3

      但是如果我们用了立即执行函数给每个 li 创造一个独立作用域,就可以改写为下面的这样,这样就能实现点击第几条就能输出几的功能。

     1 <script>
     2     var liList=document.getElementsByTagName('li');
     3     for(var i=0;i<liList.length;i++)
     4     {
     5         (function(ii) {
     6            liList[ii].onclick=function(){
     7                console.log(ii);
     8            }
     9        })(i)
    10     };
    11 </script>

      在立即执行函数执行的时候,i 的值被赋值给 ii,此后 ii 的值一直不变,如图6所示。i 的值从 0 变化到 3,对应3 个立即执行函数,这 3个立即执行函数里面的 ii 「分别」是 0、1、2。

    图6 点击第几个li,就输出几

       其实ES6语法中的let也可以实现上述的功能,仅仅是将for循环中的var换成let,如下所示,有木有觉得很简单明了。

    1 <script>
    2      var liList=document.getElementsByTagName('li');
    3      for(let i=0;i<liList.length;i++)
    4      {
    5             liList[i].onclick=function(){
    6                 console.log(i);
    7              }
    8      }
    9 </script>

      那很多人就觉得用let可以完全取代立即执行函数,到目前为止,可能是我眼界所限制,我所能用到的立即执行函数的确能被let替代,前提是你的运行环境(包括旧的浏览器)支持ES2015。如果不支持,你将不得不求助于以前经典的函数。

  • 相关阅读:
    十的次方——挖掘并非显而易见的观点与想法
    6-3-5式脑力接龙
    三个臭皮匠,顶上一个诸葛亮——在Google Ideathon上Design Thinking分享
    上瘾:如何打造习惯养成中的产品(投资篇)
    上瘾:如何打造习惯养成中的产品(奖励篇)
    上瘾:如何打造习惯养成中的产品(行动篇)
    上瘾:如何打造习惯养成中的产品(触发器篇)
    上瘾:如何打造习惯养成中的产品
    告别2013拥抱2014
    Design Thinking Workshop @ Agile Tour 2013 Shanghai
  • 原文地址:https://www.cnblogs.com/jdWu-d/p/11587805.html
Copyright © 2011-2022 走看看