引文:

定义函数的方式有两种:一种是函数声明,另一种就是函数表达式

  • 关于函数声明,它的一个重要特征就是函数声明提升(function declaration hoisting),意思是在执行

代码之前会先读取函数声明。这就意味着可以把函数声明放在调用它的语句后面:

  1. sayHi();
  2. function sayHi(){
  3. alert("Hi!");
  4. }
  • 第二种创建函数的方式是使用函数表达式, 即创建一个函数并将它赋值给变量 functionName。

这种情况下创建的函数叫做匿名函数(anonymous function),因为 function 关键字后面没有标识符。
(匿名函数有时候也叫拉姆达函数。)

函数表达式与其他表达式一样,在使用前必须先赋值。以下代码会导致错误:

  1. sayHi(); //错误:函数还不存在
  2. var sayHi = function(){
  3. alert("Hi!");
  4. };

递归

递归函数是在一个函数通过名字调用自身的情况下构成的,如下所示:

  1. function factorial(num){
  2. if (num <= 1){
  3. return 1;
  4. } else {
  5. return num * factorial(num-1);
  6. }
  7. }

这是一个经典的递归阶乘函数。虽然这个函数表面看来没什么问题,但下面的代码却可能导致它出错

  1. var anotherFactorial = factorial;
  2. factorial = null;
  3. alert(anotherFactorial(4)); //出错!

以上代码先把 factorial()函数保存在变量 anotherFactorial 中,然后将 factorial 变量设置为 null,结果指向原始函数的引用只剩下一个。但在接下来调用 anotherFactorial()时,由于必须执行 factorial(),而 factorial 已经不再是函数,所以就会导致错误。在这种情况下,使用 arguments.callee 可以解决这个问题。我们知道arguments.callee 是一个指向正在执行的函数的指针,因此可以用它来实现对函数的递归调用,例如:

  1. function factorial(num){
  2. if (num <= 1){
  3. return 1;
  4. } else {
  5. return num * arguments.callee(num-1);
  6. }
  7. }