尾递归与 memorize 优化

本文写于 2020 年 12 月 10 日

递归

递归是一种非常常见的算法思维,在大家刚开始学编程的时候应该就会接触到。

我们可以这么理解递归:

function 讲故事() {
从前有座山,山里有座庙; 庙里有个老和尚给小和尚讲故事; 讲的什么故事呢; 讲故事()
}

递归就是“我用我自己”。

递归的次数缺陷

但是众所周知,递归是会进行「压栈」和「弹栈」的。

因为递归是在自己里面调用自己,所以上一个函数根本没有结束的时候,我们就要再一次调用新的函数,这样在调用栈里面的函数根本没有机会出来——直到爆栈。

function foo() {
try {
return 1 + foo();
} catch(e) {
console.error(e);
return 1;
}
}

可以尝试在浏览器中运行这段代码,他会告诉你该浏览器的调用栈长度是多少,并且报错:Maximum call stack size exceeded.,超过调用栈的最大长度。

一般这个值会在一万左右浮动,根据不同的电脑、系统、浏览器呈现出不一样的结果。

可以看到这个值可能一般够用,但还是容易爆栈。这就是递归的第一个缺陷:次数有限

递归的速度缺陷

斐波那契第 n 项的值计算应该都大家都会:

const fib = (n) => (n === 0 || n === 1 ? n : fib(n - 1) + fib(n - 2));

这时候我们给他加一个计时功能(代码我就不写了),看看求每一项的值需要多少时间。

项数 时间(ms)
20 1
24 4
32 69
38 1193
40 3065
64 太慢了,出不来结果了

可以看到这个数字上涨的非常的离谱,这就是递归的第二个缺陷:速度太慢

尾递归

尾递归比递归快的条件,是在编译阶段进行了优化(Safari 实现了),如果编译器根本没有优化尾递归,那么速度和递归就没有区别了。

首先理解什么递归需要压栈和弹栈。

因为函数执行到一半,又去执行了函数,而这个新函数结束之后我们还需要用到当前函数里参数、环境……等,所以我们得记住他们,并且回来。

尾递归就是让我们不需要当前函数的环境了,直接 return 我们的答案,自然也就不需要将大量的信息进行压栈、弹栈了。

const fib = (n) => (n === 0 || n === 1 ? n : fib(n - 1) + fib(n - 2));

这段代码里面,我们需要回来之后进行相加,所以需要进行压栈弹栈。

Memorize 优化

关于递归速度太慢,我们的解决方案可以是:Memorize 优化

我们在计算第 n 项的时候,本质上是从第 0 项开始算起:

  1. fib(4) = fib(3) + fib(2)
  2. fib(3) = fib(2) + fib(1)
  3. fib(2) = fib(1) + fib(0)
  4. fib(1) = 1
  5. fib(0) = 0

我们是在弹栈的时候发生计算,那也就是倒着来:

  1. 先算 fib(0)fib(1)
  2. 再算 fib(2)
  3. 再算 fib(3)
  4. 再算 fib(4)——此时我们已经忘记了 fib(2) 的值,只知道 fib(3),所以还要再算一遍 fib(2)

由此我们可以知道在计算过程中发生了太多重复的计算

完全可以用一个哈希表存起来这些数据,第二次、第三次使用的时候直接获取结果就可以了,没有必要像第一次一样重新计算。

const memorize = (fn) => {
const cache = {};
return (n) => {
if (!cache[n]) {
cache[n] = fn(n);
}
return cache[n];
};
}; const fib = memorize((n) => (n === 0 || n === 1 ? n : fib(n - 1) + fib(n - 2)));

我们发现 fib(1024), fib(2048) 这种很大很大的数字我们都可以秒出答案

(完)

尾递归与 memorize 优化的更多相关文章

  1. 详解JavaScript调用栈、尾递归和手动优化

    调用栈(Call Stack) 调用栈(Call Stack)是一个基本的计算机概念,这里引入一个概念:栈帧. 栈帧是指为一个函数调用单独分配的那部分栈空间. 当运行的程序从当前函数调用另外一个函数时 ...

  2. Scala Tail Recursion (尾递归)

    Scala对尾递归进行了优化,甚至提供了专门的标注告诉编译器需要进行尾递归优化.不过这种优化仅限于严格的尾递归,间接递归等情况,不会被优化. 尾递归的概念 递归,大家都不陌生,一个函数直接或间接的调用 ...

  3. 尾递归与Continuation

    怎样在不消除递归的情况下防止栈溢出?(无论如何都要使用递归) 这几天恰好和朋友谈起了递归,忽然发现不少朋友对于“尾递归”的概念比较模糊,网上搜索一番也没有发现讲解地完整详细的资料,于是写了这么一篇文章 ...

  4. Scala尾递归

    递归函数应用 首先,我们来对比两个递归方法的求值步骤. 假设有方法gcd,用来计算两个数的最大公约数.下面是欧几里得算法的实现: def gcp(a: Int, b: Int): Int = if ( ...

  5. JavaScript 中的尾调用

    尾调用(Tail Call) 尾调用是函数式编程里比较重要的一个概念,它的意思是在函数的执行过程中,如果最后一个动作是一个函数的调用,即这个调用的返回值被当前函数直接返回,则称为尾调用,如下所示: f ...

  6. 第四章 函数(JavaScript:语言精粹)

    函数包含一组语句,用来指定对象的行为,其代码可以用来重复使用.   一般来说,编程就是将一组需求分解成一组函数和数据结构的技能.   概览:函数对象 | 函数字面量 | 调用 | 方法调用模式 | 函 ...

  7. ES6 入门系列 - 函数的扩展

    1函数参数的默认值 基本用法 在ES6之前,不能直接为函数的参数指定默认值,只能采用变通的方法. function log(x, y) { y = y || 'World'; console.log( ...

  8. Android ViewTreeObserver简介

    Android ViewTreeObserver简介   一.结构 public final class ViewTreeObserver extends Object java.lang.Objec ...

  9. SCIP读书笔记(1)

    这书也算是必修吧,尤其是我这种非科班人员,还是应该抽时间尽量学习一下.大致翻过一遍,习题非常多,尽力吧. ##构造过程抽象 * 为了表述认知,每种语言都提供了三种机制:基本元素:组合方式:抽象方法. ...

随机推荐

  1. 使用 Spring 有哪些方式?

    使用 Spring 有以下方式: 作为一个成熟的 Spring Web 应用程序. 作为第三方 Web 框架,使用 Spring Frameworks 中间层. 用于远程使用. 作为企业级 Java ...

  2. 【C/C++】C语言基础知识【第二版】

    基础语法 输出语句 #include <stdio.h> int main(void) { printf("-------分界线1------- \n"); print ...

  3. java中请给出一个return this的例子。

    [新手可忽略不影响继续学习]下面例子中setYear中的return this;返回了一个指向对象的指针,this.setMonth(8).setDay(20);是合法的,如果像原来的例子一样什么都不 ...

  4. HTML表格CSS美化

    效果展示 style.css html{ width: 100%; height: 100%; overflow: hidden;}body{ width: 100%; height: 100%; f ...

  5. 三种获取数据的方法fetch和ajax和axios

    一 .fetch用法 ( 本人比较喜欢fetch,代码精简,虽说目前axios比较流行,但是fetch很多大厂已经开始用fetch开始封装了, 我觉得以后fetch会取代axios和ajax ) 1. ...

  6. 前端实现导出excel

    结果: 将网页上拿到的数据导出成excel文件 实现: HTML代码 <div> <button type="button" onclick="expo ...

  7. Linux上部署net6应用

    前言 ​ .net6都出来了,作为一名.net搬砖工却一直都在windows下部署应用,还未尝试过linux环境下部署应用.参考福禄网络研发团队的相关博客,学习一下如何将应用部署到linux系统. . ...

  8. 中小学数学卷子自动生成程序--对G同学的代码分析

    前几天,在课程要求下完成了个人项目的项目工程编写,即一个中小学数学卷子自动生成程序. 程序主要功能是用户预设账户登录后可以选择等级进行对应的小中高的数学卷子对应出题生成txt文本. 本文针对partn ...

  9. OllyDbg---比较、条件跳转指令

    比较和条件跳转 CMP 比较两个操作数,相当于SUB指令,但是相减的结果不保存到第一个操作数,而是根据相减的结果来改变零标志位.当两个操作数相等时,零标志位Z置为1. 两个操作数不相等时,零标志位Z被 ...

  10. linux磁盘之回环设备

    为什么要讲回环设备,下面看下系统的采样情况 上面图有loop0,loop1等设备标记,这些是什么?看看下面fdisk -l 的结果,乱糟糟的一堆? 什么是回环设备? 上面的loop0, loop1就是 ...