Javascript学习笔记之 函数篇(一) : 函数声明和函数表达式
函数声明
functionfoo(){}
函数foo将会在整个程序执行前被hoist(提升),因此它在定义foo函数的整个scope(作用域)中都是可用的。即使在函数定义之前调用它也没问题。
foo();//Worksbecausefoowascreatedbeforethiscoderuns functionfoo(){}
因为我打算专门写篇介绍作用域的文章,所以这里就不详述了。
函数表达式
对于函数声明,函数的名称是必须的,而对于函数表达式而言则是可选的,因此,就出现了匿名函数表达式和命名函数表达式。如下:
函数声明:functionfunctionName(){ }
函数声明:functionfunctionName[可选](){ }
那么我就知道,如果没有函数名的话,一定就是函数表达式,但是对于有函数名的情况该如何判断呢?
Javascript规定如果整个函数体是作为表达式的一部分时,那么它就是函数表达式,否则即是函数声明。以下为表达式:
varfuc=foo(){}
我们再举几个极端的表达式例子:
!functionfoo(){} true&&functionfoo(){}
以上的语句这里只是为了区分函数表达式,一般不会这样写。那么用一个对比的例子来看看效果:
foo1();//foo1isnotdefined foo2();//worksbecausefoo2wascreatedbeforethiscoderuns !functionfoo1(){ alert('foo1works'); }; functionfoo2(){ alert('foo2works'); };
匿名函数表达式
varfoo=function(){};
上面的例子将一个匿名函数赋值给了变量foo。
foo;//'undefined' foo();//thisraisesaTypeError varfoo=function(){};
由于var是一个声明所以这里对变量foo进行hoist(提升),因此当程序执行时,变量foo是可调用的。
但是由于赋值语句只有在运行时才生效,所以变量foo的值为undefined。
命名函数表达式
另一个要讲到的就是命名函数的赋值。
varfoo=functionbar(){ bar();//Works }; bar();//ReferenceError
在这里,命名函数bar赋值给了变量foo,所以在函数声明外是不可见的,但在bar函数内部仍然可以调用。这是因为Javascript对命名函数处理的机制,函数的名称永远在函数内部的作用域中有效。