浅谈JavaScript的函数表达式(递归)

  递归函数,在前面的博客中已经简单的介绍了。递归函数是一个通过函数名称在函数内部调用自身的函数。如下:

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

  上面的代码,在第一行声明了一个fac函数,同时在6行调用了fac函数本身。这是一个求阶乘的递归函数。

1 var anthorfacc=fac;
2                     fac=null;
3                     anthorfacc(4);//抛出异常

  上面的代码,在第一行声明了一个变量anthorfacc,并指向fac。第2行将fac设置null,这样anthorfacc也同样为null。第3行调用的时候会抛出异常,因为在求阶乘的时候要两次调用fac函数,但是fac设置为null之后,事实上fac的引用只剩一个。此时,在函数内部调用fac会报错。在这种情况下,使用arguments.callee可以解决上面的问题。

  arguments.callee是一个指向执行函数的指针,可以用这个方法实现函数的递归调用。

 1 function fac(num){
 2                         if(num<1){
 3                             return 1;
 4                         }
 5                         else{
 6                             return num*arguments.callee(num-1);
 7                         }
 8                     }
 9                     var anthorfacc=fac;
10                     fac=null;
11                     var jie=anthorfacc(4);
12                     console.log(jie);//24

  上面的代码用arguments.callee代替了函数本身,即使将fac设置为null,依然不会报错。但是在es5中的严格模式是不允许使用callee方法,访问这个属性会报错。可以使用命名函数表达式来表达相同的结果。

 1 var fac=(function f(num){
 2                         if(num<1){
 3                             return 1;
 4                         }
 5                         else{
 6                             return num*fac(num-1);
 7                         }
 8                     });
 9                     var anthorfacc=fac;
10                     fac=null;
11                     var jie=anthorfacc(4);
12                     console.log(jie);//24

  上面的代码创建了一个名为f的命名函数表达式,然后将它的值赋给fac。即使将函数fac赋值给另外一个变量,函数f依然有效,所以递归函数依然有效。这种使用在严格模式和非严格模式都可以使用。

posted on 2018-01-30 17:21  水击三千  阅读(250)  评论(0编辑  收藏  举报

导航