JS预解析机制与定义函数的三种方式

一、 定义函数的三种方式

1、函数声明 (用的最多,推荐优先使用)

<script>
    //函数声明的语法
    function fn() {
        console.log("这是函数声明的方式");
    }
    fn(); //直接调用函数名
</script>

 

2、函数表达式 (也叫匿名函数)

<script>
    var fn = function([参数列表]) {
        console.log("这是一个匿名函数");
    /*
    顾名思义,因为这种写法是把函数赋给了一个变量,函数并没有真正的名字
    因此就叫匿名函数
    */
    fn(); //调用匿名函数,使用变量名称加();
</script>
 

3、自执行函数 (1)第一种写法

<script>
    /*
    顾名思义:自己执行自己,并且在声明的同时就调用自己,只能调用一次
    */
    //有两种写法,这是第一种:
​
    ();
</script>

 

这样写的原理是:他是由函数表达式演变过来的,var fn = function(){} fn();把fn替换成: function(){}, 就成了: function(){}();为了保持整体性,在function加上一个(),所以就变成 了(function([这里 是形参]){})([这里是实参])

(2)第二种写法

<script>
    //第二种写法:
    (function() {
        console.log("这是一个自执行函数");
    }())
</script>    

 

值得注意的是,用这种写法会有一个小漏洞 ,看个例子:

<script>
    var fn = function() {
        console.log(123);
    }
    (function() {
        console.log(456);
    }())
</script>    

 

输出结果是:

这里匿名函数fn明明没调用,为什么会输出?看下面简化后的例子:

<script>
    var fn = function() {
        console.log(123);
    };
    (function() {
        console.log(456);
    }())
</script>

 

可以很明显的看到(function(){console.log(456);}())这个自执行函数调用输出了456,同 时,这个自执行函数跟在匿名函数的后面,JS解析的时候把这个匿名函数也解析成为了自 执行函数,所以就输出了123 ,解决的办法很简单,在第一个匿名函数的后面加一 个分号";" 或者直接采取第一种写法就能避免这种问题.

二、JS预解析机制

1、预解析过程: (1 )把变量的声明提升到当前作用域的最前面,只会提升声明,不会提升赋值。 ( 2 )把函数的声明提升到当前作用域的最前面,只会提升声明,不会提升调用。 (3 )先提升var,在提升function 因此,我们在定义函数时,绝大多数都使用第一种方式(函数声明的方式) 匿名函数多用于定时器里面和注册事件的时候,比如

<script>
    btn.onclick = function() {
        console.log("这是一个匿名函数");
    }
</script>

 

2、预解析示例:

<script>
    var num= 789;
    function fn(){
        console.log(num);
        var num = 789;
    }
    fn(); //调用fn函数
</script>

 

此时控制台打印的是undefined , 原因是: JS解析代码时,把函数的声明还有变量的声明 提升到当前作用域的最前面,所以代码就变成:

<script>
    var num;
    num = 789;
    function fn(){
        var num;//从这里可以看出num只是声明而没有定义赋值
        console.log(num); //输出的时候必然是undefined
        num = 789;
    }
    fn();
</script>

 

 

posted @ 2020-05-25 10:38  hello9102  阅读(194)  评论(0编辑  收藏  举报