尾调用(Tail Call)是函数式编程的一个重要概念,自己很是简单,一句话就能说清楚,就是指某个函数的最后一步是调用另外一个函数。编程
//尾调用 function f(x) {return g(x);} //上面代码中,函数f的最后一步是调用函数g,这就叫尾调用 //不属于尾调用举例 // 状况一 function f(x){ let y = g(x); return y; } // 状况二 function f(x){ return g(x) + 1; } // 状况三 function f(x){ g(x); }
尾调用之因此与其余调用不一样,就在于它的特殊的调用位置。函数式编程
咱们知道,函数调用会在内存造成一个“调用记录”,又称“调用帧”(call frame),保存调用位置和内部变量等信息。若是在函数A的内部调用函数B,那么在A的调用帧上方,还会造成一个B的调用帧。等到B运行结束,将结果返回到A,B的调用帧才会消失。若是函数B内部还调用函数C,那就还有一个C的调用帧,以此类推。全部的调用帧,就造成一个“调用栈”(call stack)。函数
尾调用因为是函数的最后一步操做,因此不须要保留外层函数的调用帧,由于调用位置、内部变量等信息都不会再用到了,只要直接用内层函数的调用帧,取代外层函数的调用帧就能够了。优化
function f() { let m = 1; let n = 2; return g(m + n); } f(); // 等同于 function f() { return g(3); } f(); // 等同于 g(3);
上面代码中,若是函数g不是尾调用,函数f就须要保存内部变量m和n的值、g的调用位置等信息。但因为调用g以后,函数f就结束了,因此执行到最后一步,彻底能够删除 f(x) 的调用帧,只保留 g(3) 的调用帧。this
这就叫作“尾调用优化”(Tail call optimization),即只保留内层函数的调用帧。若是全部函数都是尾调用,那么彻底能够作到每次执行时,调用帧只有一项,这将大大节省内存。这就是“尾调用优化”的意义。spa
注意,只有再也不用到外层函数的内部变量,内层函数的调用帧才会取代外层函数的调用帧,不然就没法进行“尾调用优化”。code
函数调用自身,称为递归。若是尾调用自身,就称为尾递归。blog
递归很是耗费内存,由于须要同时保存成千上百个调用帧,很容易发生“栈溢出”错误(stack overflow)。但对于尾递归来讲,因为只存在一个调用帧,因此永远不会发生“栈溢出”错误。 以斐波那契数列(Fibonacci)为例分析:递归
//普通递归 function Fibonacci (n) { if ( n <= 1 ) {return 1}; return Fibonacci(n - 1) + Fibonacci(n - 2); } Fibonacci(10); // 89 // Fibonacci(100) // Fibonacci(500) // 堆栈溢出了 //尾递归 function Fibonacci2 (n , ac1 = 1 , ac2 = 1) { if( n <= 1 ) {return ac2}; return Fibonacci2 (n - 1, ac2, ac1 + ac2); //保留一个调用记录,复杂度O(1) } Fibonacci2(100) // 573147844013817200000 Fibonacci2(1000) // 7.0330367711422765e+208 Fibonacci2(10000) // Infinity
尾递归的实现,每每须要改写递归函数,确保最后一步只调用自身。作到这一点的方法,就是把全部用到的内部变量改写成函数的参数。这样作的缺点是不太直观,解决办法有两种:内存
方法1:在尾递归函数以外,再提供一个正常形式的函数。
//尾递归 function Fibonacci2 (n , ac1 = 1 , ac2 = 1) { if( n <= 1 ) {return ac2}; return Fibonacci2 (n - 1, ac2, ac1 + ac2); } Fibonacci2(100) // 573147844013817200000 Fibonacci2(1000) // 7.0330367711422765e+208 Fibonacci2(10000) // Infinity //优化1 function tailFibonacci(n, ac1, ac2){ if( n <= 1 ) {return ac2}; return tailFibonacci (n - 1, ac2, ac1 + ac2); } function Fibonacci3(n){ return tailFibonacci(n, 1 , 1); } Fibonacci3(100) // 573147844013817200000
方法2:函数式编程有一个概念,叫作柯里化(currying),意思是将多参数的函数转换成单参数的形式。这里也可使用柯里化。
//优化2 柯里化 function currying(fn, n1, n2) { return function (m) { return fn.call(this, m, n1, n2); }; } function tailFibonacci(n, ac1, ac2){ if( n <= 1 ) {return ac2}; return tailFibonacci (n - 1, ac2, ac1 + ac2); } const Fibonacci4 = currying(tailFibonacci,1,1); Fibonacci4(100) // 573147844013817200000