在函数内部,有两个特殊的对象:arguments 和 this。其中, arguments 的主要用途是保存函数参数, 但这个对象还有一个名叫 callee 的属性,该属性是一个指针,指向拥有这个 arguments 对象的函数。 请看下面这个很是经典的阶乘函数javascript
function factorial(num){ if (num <=1) { return 1; } else { return num * factorial(num-1) } }
定义阶乘函数通常都要用到递归算法;如上面的代码所示,在函数有名字,并且名字之后也不会变 的状况下,这样定义没有问题。但问题是这个函数的执行与函数名 factorial 牢牢耦合在了一块儿。为 了消除这种紧密耦合的现象,能够像下面这样使用 arguments.calleejava
function factorial(num){ if (num <=1) { return 1; } else { return num * arguments.callee(num-1); } }
在这个重写后的 factorial()函数的函数体内,没有再引用函数名 factorial。这样,不管引用 函数时使用的是什么名字,均可以保证正常完成递归调用。例如面试
function factorial(num){ if(num <= 1){ return 1; }else{ return num * arguments.callee(num-1); } } var trueFactorial = factorial; alert(trueFactorial(5)); //120
factorial = function() { return 0; } alert(trueFactorial(5));// 120 若是没有使用arguments.callee,将返回0
在此,变量 trueFactorial 得到了 factorial 的值,其实是在另外一个位置上保存了一个函数 的指针。而后,咱们又将一个简单地返回 0的函数赋值给 factorial 变量。若是像原来的 factorial() 那样不使用 arguments.callee,调用 trueFactorial()就会返回 0。但是,在解除了函数体内的代 码与函数名的耦合状态以后,trueFactorial()仍然可以正常地计算阶乘;至于 factorial(),它现 在只是一个返回 0的函数。 算法
参考自js高程第三版数组
2017/4/5号更新浏览器
如今已经不推荐使用arguments.callee();闭包
缘由:访问 arguments 是个很昂贵的操做,由于它是个很大的对象,每次递归调用时都须要从新建立。影响现代浏览器的性能,还会影响闭包。函数
不能用怎么办?性能
像第三段中的例子,重写 factorial()方法致使trueFactorial()结果不在预期。是为了演示而作的。平时写代码应该避免。this
递归时用到arguments.callee()是常见的事情,好比
一道面试题。接受参数n=5,不用for循环输出数组【1,2,3,4,5】
这用递归的思路,配合arguments.callee,代码以下
function show(n) { var arr = []; return (function () { arr.unshift(n); n--; if (n != 0) { arguments.callee(); } return arr; })() }
show(5)//[1,2,3,4,5]
如今arguments.callee 被弃用了。怎么办,其实很简单,给内部函数一个名字便可
function show(n) { var arr = []; return (function fn() { arr.unshift(n); n--; if (n != 0) { fn(); } return arr; })() }show(5)//[1,2,3,4,5]