JavaScript 数据结构与算法之美 - 递归

在线体验各类最新模型,更有模型 免费Token 额度领取!
立即体验
简介: 算法为王。排序算法博大精深,前辈们用了数年甚至一辈子的心血研究出来的算法,更值得我们学习与推敲。因为之后要讲有内容和算法,其代码的实现都要用到递归,所以,搞懂递归非常重要。

1. 定义


  • 方法或函数调用自身的方式称为递归调用,调用称为递,返回称为归。

简单来说就是:自己调用自己


现实例子:周末你带着女朋友去电影院看电影,女朋友问你,咱们现在坐在第几排啊 ?电影院里面太黑了,看不清,没法数,现在你怎么办 ?


于是你就问前面一排的人他是第几排,你想只要在他的数字上加一,就知道自己在哪一排了。


但是,前面的人也看不清啊,所以他也问他前面的人。


就这样一排一排往前问,直到问到第一排的人,说我在第一排,然后再这样一排一排再把数字传回来。


直到你前面的人告诉你他在哪一排,于是你就知道答案了。


基本上,所有的递归问题都可以用递推公式来表示,比如:


f(n) = f(n-1) + 1; 
// 其中,f(1) = 1


f(n) 表示你想知道自己在哪一排,f(n-1) 表示前面一排所在的排数,f(1) = 1 表示第一排的人知道自己在第一排。


有了这个递推公式,我们就可以很轻松地将它改为递归代码,如下:


function f(n) {
  if (n == 1) return 1;
  return f(n-1) + 1;
}


2. 为什么使用递归 ?递归的优缺点 ?


  • 优点:代码的表达力很强,写起来简洁。
  • 缺点:空间复杂度高、有堆栈溢出风险、存在重复计算、过多的函数调用会耗时较多等问题。


3. 什么样的问题可以用递归解决呢 ?


一个问题只要同时满足以下 3 个条件,就可以用递归来解决。


  1. 问题的解可以分解为几个子问题的解。何为子问题 ?就是数据规模更小的问题。


比如,前面讲的电影院的例子,你要知道,自己在哪一排的问题,可以分解为前一排的人在哪一排这样一个子问题。


  1. 问题与子问题,除了数据规模不同,求解思路完全一样

比如电影院那个例子,你求解自己在哪一排的思路,和前面一排人求解自己在哪一排的思路,是一模一样的。


  1. 存在递归终止条件

比如电影院的例子,第一排的人不需要再继续询问任何人,就知道自己在哪一排,也就是 f(1) = 1,这就是递归的终止条件。


4. 递归常见问题及解决方案


  1. 警惕堆栈溢出:可以声明一个全局变量来控制递归的深度,从而避免堆栈溢出。
  2. 警惕重复计算:通过某种数据结构来保存已经求解过的值,从而避免重复计算。


5. 如何实现递归 ?


1. 递归代码编写


写递归代码的关键就是找到如何将大问题分解为小问题的规律,并且基于此写出递推公式,然后再推敲终止条件,最后将递推公式和终止条件翻译成代码。


2. 递归代码理解


对于递归代码,若试图想清楚整个递和归的过程,实际上是进入了一个思维误区。

那该如何理解递归代码呢 ?


  • 如果一个问题 A 可以分解为若干个子问题 B、C、D,你可以假设子问题 B、C、D 已经解决。
  • 而且,你只需要思考问题 A 与子问题 B、C、D 两层之间的关系即可,不需要一层层往下思考子问题与子子问题,子子问题与子子子问题之间的关系。
  • 屏蔽掉递归细节,这样子理解起来就简单多了。


因此,理解递归代码,就把它抽象成一个递推公式,不用想一层层的调用关系,不要试图用人脑去分解递归的每个步骤。


6. 例子


1. 一个阶乘的例子:


function fact(num) {
  if (num <= 1) {
    return 1;
  } else {
    return num * fact(num - 1);
    }
}
fact(3) // 结果为 6
以下代码可导致出错:
var anotherFact = fact; 
fact = null; 
alert(antherFact(4)); //出错


由于 fact 已经不是函数了,所以出错。


使用 arguments.callee


arguments.callee 是一个指向正在执行的函数的指针,arguments.callee 返回正在被执行的对现象。


新的函数为:


function fact(num){ 
    if (num <= 1){ 
        return 1; 
    }else{ 
        return num * arguments.callee(num - 1); //此处更改了。 
    } 
} 
var anotherFact = fact; 
fact = null; 
alert(antherFact(4)); // 结果为 24


2. 再看一个多叉树的例子


先看图


微信图片_20220513122105.png


叶子结点:就是深度为 0 的结点,也就是没有孩子结点的结点,简单的说就是一个二叉树任意一个分支上的终端节点。


数据结构格式,参考如下代码:


const json = {
  name: 'A',
  children: [
    {
      name: 'B',
      children: [
        {
          name: 'E',
        },
        {
          name: 'F',
        },
        {
          name: 'G',
        }
      ]
    },
    {
      name: 'C',
      children: [
        {
          name: 'H'
        }
      ]
    },
    {
      name: 'D',
      children: [
        {
          name: 'I',
        },
        {
          name: 'J',
        }
      ]
    }
  ]
}


我们如何获取根节点的所有叶子节点个数呢 ?


递归代码如下:


/**
 * 获取根节点的所有 叶子节点 个数
 * @param {Object} json Object 对象
 */
function getLeafCountTree(json) {
  if(!json.children){
      return 1;
  } else {
      let leafCount = 0;
      for(let i = 0 ; i < json.children.length ; i++){
          // leafCount = leafCount + getLeafCountTree(json.children[i]);
          leafCount = leafCount + arguments.callee(json.children[i]);
      }
      return leafCount;
  }
}


递归遍历是比较常用的方法,比如:省市区遍历成树、多叉树、阶乘等。


7. 文章输出计划


JavaScript 数据结构与算法之美 的系列文章,坚持 3 - 7 天左右更新一篇,暂定计划如下表。


| 标题 | 链接 |

| :------ | :------ |

| 时间和空间复杂度 | https://github.com/biaochenxu... |

| 线性表(数组、链表、栈、队列) | https://github.com/biaochenxu... |

| 实现一个前端路由,如何实现浏览器的前进与后退 ?| https://github.com/biaochenxu... |

| 栈内存与堆内存 、浅拷贝与深拷贝 | https://github.com/biaochenxu... |

| 递归 | https://github.com/biaochenxu... |

| 非线性表(树、堆) | 精彩待续 |

| 冒泡排序 | 精彩待续 |

| 插入排序 | 精彩待续 |

| 选择排序 | 精彩待续 |

| 归并排序 | 精彩待续 |

| 快速排序 | 精彩待续 |

| 计数排序 | 精彩待续 |

| 基数排序 | 精彩待续 |

| 桶排序 | 精彩待续 |

| 希尔排序 | 精彩待续 |

| 堆排序 | 精彩待续 |

| 十大经典排序汇总 | 精彩待续 |

如果有错误或者不严谨的地方,请务必给予指正,十分感谢。
相关文章
|
9月前
|
存储 监控 算法
局域网监控其他电脑的设备信息管理 Node.js 跳表算法
跳表通过分层索引实现O(logn)的高效查询、插入与删除,适配局域网监控中设备动态接入、IP映射及范围筛选等需求,相比传统结构更高效稳定,适用于Node.js环境下的实时设备管理。
300 9
|
11月前
|
存储 监控 JavaScript
基于布隆过滤器的 Node.js 算法在局域网电脑桌面监控设备快速校验中的应用研究
本文探讨了布隆过滤器在局域网电脑桌面监控中的应用,分析其高效空间利用率、快速查询性能及动态扩容优势,并设计了基于MAC地址的校验模型,提供Node.js实现代码,适用于设备准入控制与重复数据过滤场景。
348 0
|
运维 监控 JavaScript
内网网管软件中基于 Node.js 的深度优先搜索算法剖析
内网网管软件在企业网络中不可或缺,涵盖设备管理、流量监控和安全防护。本文基于Node.js实现深度优先搜索(DFS)算法,解析其在网络拓扑遍历中的应用。通过DFS,可高效获取内网设备连接关系,助力故障排查与网络规划。代码示例展示了图结构的构建及DFS的具体实现,为内网管理提供技术支持。
288 11
|
9月前
|
存储 监控 JavaScript
企业上网监控系统的恶意 URL 过滤 Node.js 布隆过滤器算法
布隆过滤器以低内存、高效率特性,解决企业上网监控系统对百万级恶意URL实时检测与动态更新的难题,通过概率性判断实现毫秒级过滤,内存占用降低96%,适配大规模场景需求。
444 3
|
9月前
|
存储 监控 算法
电脑管控软件的进程优先级调度:Node.js 红黑树算法
红黑树凭借O(log n)高效插入、删除与查询特性,适配电脑管控软件对进程优先级动态调度的高并发需求。其自平衡机制保障系统稳定,低内存占用满足轻量化部署,显著优于传统数组或链表方案,是实现关键进程资源优先分配的理想选择。
389 1
|
10月前
|
运维 监控 JavaScript
基于 Node.js 图结构的局域网设备拓扑分析算法在局域网内监控软件中的应用研究
本文探讨图结构在局域网监控系统中的应用,通过Node.js实现设备拓扑建模、路径分析与故障定位,提升网络可视化、可追溯性与运维效率,结合模拟实验验证其高效性与准确性。
507 3
|
Web App开发 数据采集 JavaScript
动态网页爬取:Python如何获取JS加载的数据?
动态网页爬取:Python如何获取JS加载的数据?
1874 58
|
机器学习/深度学习 JavaScript 前端开发
JS进阶教程:递归函数原理与篇例解析
通过对这些代码示例的学习,我们已经了解了递归的原理以及递归在JS中的应用方法。递归虽然有着理论升华,但弄清它的核心思想并不难。举个随手可见的例子,火影鸣人做的影分身,你看到的都是同一个鸣人,但他们的行为却能在全局产生影响,这不就是递归吗?雾里看花,透过其间你或许已经深入了递归的魅力之中。
445 19
|
监控 算法 JavaScript
基于 JavaScript 图算法的局域网网络访问控制模型构建及局域网禁止上网软件的技术实现路径研究
本文探讨局域网网络访问控制软件的技术框架,将其核心功能映射为图论模型,通过节点与边表示终端设备及访问关系。以JavaScript实现DFS算法,模拟访问权限判断,优化动态策略更新与多层级访问控制。结合流量监控数据,提升网络安全响应能力,为企业自主研发提供理论支持,推动智能化演进,助力数字化管理。
335 4
|
监控 算法 JavaScript
公司局域网管理视域下 Node.js 图算法的深度应用研究:拓扑结构建模与流量优化策略探析
本文探讨了图论算法在公司局域网管理中的应用,针对设备互联复杂、流量调度低效及安全监控困难等问题,提出基于图论的解决方案。通过节点与边建模局域网拓扑结构,利用DFS/BFS实现设备快速发现,Dijkstra算法优化流量路径,社区检测算法识别安全风险。结合WorkWin软件实例,展示了算法在设备管理、流量调度与安全监控中的价值,为智能化局域网管理提供了理论与实践指导。
350 3

热门文章

最新文章