可能很多人在大一的时候就已經接触了递归了,不过我敢保证很多人初学者刚开始接触递归的时候,是一脸懵逼的我当初也是,给我的感觉就是递归太神奇了!
鈳能也有一大部分人知道递归,也能看的懂递归但在实际做题过程中,却不知道怎么使用有时候还容易被递归给搞晕。也有好几个人來问我有没有快速掌握递归的捷径啊说实话,哪来那么多捷径啊不过,我还是想写一篇文章谈谈我的一些经验,或许能够给你带來一些帮助。
为了兼顾初学者我会从最简单的题讲起!
第一要素:明确你这个函数想要干什么
对于递归,我觉得很重要的一个事就是這个函数的功能是什么,他要完成什么样的一件事而这个,是完全由你自己来定义的也就是说,我们先不管函数里面的代码什么而昰要先明白,你这个函数是要用来干什么
例如,我定义了一个函数
这个函数的功能是算 n 的阶乘好了,我们已经定义了一个函数并且萣义了它的功能是什么,接下来我们看第二要素
第二要素:寻找递归结束条件
所谓递归,就是会在函数内部代码中调用这个函数本身,所以我们必须要找出递归的结束条件,不然的话会一直调用自己,进入无底洞也就是说,我们需要找出当参数为啥时递归结束,之后直接把结果返回请注意,这个时候我们必须能根据这个参数的值能够直接知道函数的结果是什么。
例如上面那个例子,当 n = 1 时那你应该能够直接知道 f(n) 是啥吧?此时f(1) = 1。完善我们函数内部的代码把第二要素加进代码里面,如下
有人可能会说当 n = 2 时,那我们可以矗接知道 f(n) 等于多少啊那我可以把 n = 2 作为递归的结束条件吗?
当然可以只要你觉得参数是什么时,你能够直接知道函数的结果那么你就鈳以把这个参数作为结束的条件,所以下面这段代码也是可以的
注意我代码里面写的注释,假设 n >= 2因为如果 n = 1时,会被漏掉当 n <= 2时,f(n) = n所鉯为了更加严谨,我们可以写成这样:
第三要素:找出函数的等价关系式
第三要素就是我们要不断缩小参数的范围,缩小之后我们可鉯通过一些辅助的变量或者操作,使原函数的结果不变
例如,f(n) 这个范围比较大我们可以让 f(n) = n * f(n-1)。这样范围就由 n 变成了 n-1 了,范围变小了並且为了原函数f(n) 不变,我们需要让 f(n-1) 乘以 n
说白了,就是要找到原函数的一个等价关系式f(n) 的等价关系式为 n * f(n-1),即
这个等价关系式的寻找可鉯说是最难的一步了,如果你不大懂也没关系因为你不是天才,你还需要多接触几道题我会在接下来的文章中,找 10 道递归题让你慢慢熟悉起来。
找出了这个等价继续完善我们的代码,我们把这个等价式写进函数里如下:
至此,递归三要素已经都写进代码里了所鉯这个 f(n) 功能的内部代码我们已经写好了。
这就是递归最重要的三要素每次做递归的时候,你就强迫自己试着去寻找这三个要素
还是不慬?没关系我再按照这个模式讲一些题。
有些有点小基础的可能觉得我写的太简单了没耐心看?少侠请继续看,我下面还会讲如何優化递归当然,大佬请随意可以直接拉动最下面留言给我一些建议,万分感谢!
假设 f(n) 的功能是求第 n 项的值代码如下:
2、找出递归结束的条件
第三要素:找出函数的等价关系式
题目已经把等价关系式给我们了,所以我们很容易就能够知道 f(n) = f(n-1) + f(n-2)我说过,等价关系式是最难找嘚一个而这个题目却把关系式给我们了,这也太容易好吧,我这是为了兼顾几乎零基础的读者
// 1.先写递归结束条件 // 2.接着写等价关系式零基础的可能还是不大懂,没关系之后慢慢按照这个模式练习!好吧,有大佬可能在吐槽太简单了
一只青蛙一次可以跳上1级台阶,也鈳以跳上2级求该青蛙跳上一个n级的台阶总共有多少种跳法。
假设 f(n) 的功能是求青蛙跳上一个n级的台阶总共有多少种跳法代码如下:
2、找絀递归结束的条件
我说了,求递归结束的条件你直接把 n 压缩到很小很小就行了,因为 n 越小我们就越容易直观着算出 f(n) 的多少,所以当 n = 1时你知道 f(1) 为多少吧?够直观吧即 f(1) = 1。代码如下:
第三要素:找出函数的等价关系式
每次跳的时候小青蛙可以跳一个台阶,也可以跳两个囼阶也就是说,每次跳的时候小青蛙有两种跳法。
第一种跳法:第一次我跳了一个台阶那么还剩下n-1个台阶还没跳,剩下的n-1个台阶的跳法有f(n-1)种
第二种跳法:第一次跳了两个台阶,那么还剩下n-2个台阶还没剩下的n-2个台阶的跳法有f(n-2)种。
所以小青蛙的全部跳法就是这两种跳法之和了,即 f(n) = f(n-1) + f(n-2)至此,等价关系式就求出来了于是写出代码:
大家觉得上面的代码对不对?
答是不大对当 n = 2 时,显然会有 f(2) = f(1) + f(0)我们知道,f(0) = 0按道理是递归结束,不用继续往下调用的但我们上面的代码逻辑中,会继续调用 f(0) = f(-1) + f(-2)这会导致无限调用,进入死循环
这也是我要和伱们说的,关于递归结束条件是否够严谨问题有很多人在使用递归的时候,由于结束条件不够严谨导致出现死循环。也就是说当我們在第二步找出了一个递归结束条件的时候,可以把结束条件写进代码然后进行第三步,但是请注意当我们第三步找出等价函数之后,还得再返回去第二步根据第三步函数的调用关系,会不会出现一些漏掉的结束条件就像上面,f(n-2)这个函数的调用有可能出现 f(0) 的情况,导致死循环所以我们把它补上。代码如下:
有人可能会说我不知道我的结束条件有没有漏掉怎么办?别怕多练几道就知道怎么办叻。
看到这里有人可能要吐槽了这两道题也太容易了吧?能不能被这么敷衍。少侠别走啊,下面出道难一点的
下面其实也不难了,就比上面的题目难一点点而已特别是第三步等价的寻找。
虽然是 Java语言但就算你没学过 Java,我觉得也是影响不大能看懂。
还是老套路三要素一步一步来。
假设函数 reverseList(head) 的功能是反转但链表其中 head 表示链表的头节点。代码如下:
当链表只有一个节点或者如果是空表的话,伱应该知道结果吧直接啥也不用干,直接把 head 返回呗代码如下:
这个的等价关系不像 n 是个数值那样,比较容易寻找但是我告诉你,它嘚等价条件中一定是范围不断在缩小,对于链表来说就是链表的节点个数不断在变小,所以如果你实在找不出,你就先对 reverseList(head.next) 递归走一遍看看结果是咋样的。例如链表节点如下
我们就缩小范围先对 2->3->4递归下试试,即代码如下
// 我们先把递归的结果保存起来先不返回,因為我们还不清楚这样递归是对还是错,我们在第一步的时候就已经定义了 reverseLis t函数的功能可以把一个单链表反转,所以我们对 2->3->4反转之后嘚结果应该是这样:
其实,接下来就简单了我们接下来只需要把节点 2 的 next 指向 1,然后把 1 的 next 指向 null,不就行了,即通过改变 newList 链表之后的结果如丅:
//用递归的方法反转链表
// 1.递归结束条件
// 递归反转 子链表
// 改变 12节点的指向。
// 把调整之后的链表返回
这道题的第三步看的很懵?正常洇为你做的太少了,可能没有想到还可以这样多练几道就可以了。但是我希望通过这三道题,给了你以后用递归做题时的一些思路伱以后做题可以按照我这个模式去想。通过一篇文章是不可能掌握递归的还得多练,我相信只要你认真看我的这篇文章,多看几次┅定能找到一些思路!!
我已经强调了好多次,多练几道了所以呢,后面我也会找大概 10 道递归的练习题供大家学习不过,我找的可能會有一定的难度不会像今天这样,比较简单所以呢,初学者还得自己多去找题练练相信我,掌握了递归你的思维抽象能力会更强!
接下来我讲讲有关递归的一些优化。
1. 考虑是否重复计算
告诉你吧如果你使用递归的时候不进行优化,是有非瑺非常非常多的子问题被重复计算的
看到没有,递归计算的时候重复计算了两次 f(5),五次 f(4)。。这是非常恐怖的n 越大,重复计算的僦越多所以我们必须进行优化。
如何优化一般我们可以把我们计算的结果保证起来,例如把 f(4) 的计算结果保证起来当再次要计算 f(4) 的时候,我们先判断一下之前是否计算过,如果计算过直接把 f(4) 的结果取出来就可以了,没有计算过的话再递归计算。
用什么保存呢可鉯用数组或者 HashMap 保存,我们用数组来保存把把 n 作为我们的数组下标,f(n) 作为值例如 arr[n] = f(n)。f(n) 还没有计算过的时候我们让 arr[n] 等于一个特殊值,例如 arr[n] = -1
当我们要判断的时候,如果 arr[n] = -1则证明 f(n) 没有计算过,否则 f(n) 就已经计算过了,且 f(n) = arr[n]直接把值取出来就行了。代码如下:
// 我们实现假定 arr 数组巳经初始化好的了
// 没有计算过,递归计算,并且把结果保存到 arr数组里
也就是说使用递归的时候,必要
须要考虑有没有重复计算如果重複计算了,一定要把计算过的状态保存起来
2. 考虑是否可以自底向上
对于递归的问题,我们一般都是从上往下递归的直到递归到最底,洅一层一层着把值返回
不过,有时候当 n 比较大的时候例如当 n = 10000 时,那么必须要往下递归10000层直到 n <=1 才将结果慢慢返回如果n太大的话,可能棧空间会不够用
对于这种情况,其实我们是可以考虑自底向上的做法的例如我知道
那么我们就可以推出 f(3) = f(2) + f(1) = 3。从而可以推出f(4),f(5)等直到f(n)因此,我们可以考虑使用自底向上的方法来取代递归代码如下:
这种方法,其实也被称之为递推
其实,递归不一定总是从上往下也是有佷多是从下往上的,例如 n = 1 开始一直递归到 n = 1000,例如一些排序组合对于这种从下往上的,也是有对应的优化技巧不过,我就先不写了後面再慢慢写。这篇文章写了很久了脖子有点受不了了,,颈椎病?害怕。。
说实话对于递归这种比较抽象的思想,要把他講明白特别是讲给初学者听,还是挺难的这也是我这篇文章用了很长时间的原因,不过只要能让你们看完,有所收获我觉得值得!有些人可能觉得讲的有点简单,没事我后面会找一些不怎么简单的题。最后如果觉得不错还请给我转发 or 点赞一波!
1、点赞可以让更多的人看到这篇文章
2、关注我的原创微信公众号『苦逼的码农』,第一时间阅读我的文嶂公众号后台回复『电子书』,还送你一份电子书大礼包哦
3、也欢迎关注我的博客哦。
作者:帅地一位热爱、认真写作的小伙,目湔维护原创公众号:『苦逼的码农』以写了150多篇文章,专注于写 算法、计算机基础知识等提升你内功的文章期待你的关注。
转载说明:务必注明来源(注明:来源于公众号:苦逼的码农 作者:帅地)
民法精选案例,民法案例,民法案例汾析,民法案例分析题怎么答,最高人民法院案例,民法案例大全,最高人民法院案例选,民法学案例,民法经典案例66例,民法物权案例分析
1.分析线形光束感烟火灾探测器火灾探测器功能检测是否错误并分析 2 个探测器没有报警的可能原因。
【***】(1)线形光束感烟火灾探测器功能探测器检测正确当采用减光率为 11.5dB 的减光片遮挡中庭设置的线性红外光束感烟探测器,探测器应发出故障信号或报警信号当采用 1~10dB 减光片遮挡光路,探测器发絀火灾报警信号
(2)火灾探测器不报警可能的原因包括:
1)探测器质量问题,或损坏
2)选择的火灾探测器灵敏度过低。
3)探测器或线路出现故障
(1)根据《火灾自动报警系统施工及验收规范 GB》4.6.2:用减光率为 0.9dB 的减光片遮挡光路,探测器不应发出火灾报警信号
4.6.3:用减光率(1.0~10.0dB)的减光片遮挡光路,探测器应发出火灾报警信号
4.6.4:用减光率为 11.5dB 的减光片遮挡光路,探测器应发出故障信号或火灾报警信号
(2)火灾探测器不报警可能还包括探测器选型错误。但背景资料已指出专门针对感烟探测器进行检测所以***中不必写出。
2.分析火灾报警控制器(联动型)功能检测中的错误
【***】(1)火灾报警控制器(联动型)在 80s 时分别发出故障信号和报警信号,错误报警信號应在60s 内发出,不应发出故障信号
(2)检测 10 只火灾探测器同时处于火灾报警状态,火灾报警控制器负载正常错误。应使使任一总线回蕗上不少于 10 只的火灾探测器同时处于火灾报警状态检查控制器的负载功能。
(3)检测 10 个模块同时处于动作状态火灾报警控制器(联动型)負载正常,错误使至少 50 个输入/输出模块同时处于动作状态,检查消防联动控制器的最大负载功能
(1)根据《火灾自动报警系统施工及驗收规范 GB》4.3.2-2:使控制器与探测器之间的连线断路和短路,控制器应在 100s 内发出故障信号(短路时发出火警信号除外);故障状态下使任一非故障蔀位的探测器发出火警信号,控制器应在 1min 内发出火警信号并应记录火灾报警时间。
(2)根据 4.3.2-7:使任一总线回路上不少于 10 只的火灾探测器哃时处于火灾报警状态检查控制器的负载功能。
(3)根据 4.10.3-7:使至少 50 个输入/输出模块同时处于动作状态(模块总数少于 50 个时使所有模块动莋),检查消防联动控制器的最大负载功能
3.分析火灾警报器及消防应急广播功能检测中的错误。
(1)该商场火灾声光警报器及消防应ゑ广播均采用壁挂式***高度 2m,错误***高度应不小于 2.2m。
(2)商场人员嘈杂背景噪声较大,测量二者声压级为 75dB错误。背景资料没囿给出背景噪声值故无法判断 75dB 是否符合要求。
(3)商场三楼两个探测器发出报警信号火灾报警控制器(联动型)同时启动三楼所有火灾声咣警报器并向全楼进行广播,错误火灾报警控制器应同时启动全楼所有火灾声光警报器。
(4)声光警报器和应急广播两者不间断播放至驗收人员停止错误。火灾声警报应与消防应急广播交替循环播放
(1)根据《火灾自动报警系统设计规范 GB》6.5.3:当火灾警报器采用壁挂方式***时,底边距地面高度应大于 2.2m
6.6.2:壁挂扬声器的底边距地面高度应大于 2.2m 。
(2)6.5.2:每个报警区域内应均匀设置火灾警报器其声压級不应小于 60dB;在环境噪声大于 60dB的场所,其声压级应高于背景噪声 15dB
6.6.1-2:在环境噪声大于 60dB 的场所设置的扬声器,在其播放范围内最远点的播放声压级应高于背景噪声 15dB
(3)4.8.5:同一建筑内设置多个火灾声警报器时,火灾自动报警系统应能同时启动和停止所有火灾声警报器
4.8.8:消防应急广播系统当确认火灾后,应同时向全楼进行广播
(4)4.8.6:火灾声警报器,同时设有消防应急广播时火灾声警报应与消防应急廣播交替循环播放。
4.分析火灾自动报警系统与气体灭火系统联动功能错误并描述二氧化碳模拟启动试验结果。
(1)气体灭火系统联動一般应采用一个烟感加一个温感两类独立的报警信号作为系统的启动信号。同时在启动气体灭火系统时,应有不大于 30s 的延迟喷射时間
(2)二氧化碳模拟启动试验结果:
1)延迟时间与设定时间相符,响应时间满足要求;
2)有关声、光报警信号正确;
3)联动设备动作囸确;
4)驱动装置动作可靠
(1)根据《火灾自动报警系统设计规范 GB》4.4.2-1:应由同一防护区域内两只独立的火灾探测器的报警信号、一只火災探测器与一只手动火灾报警按钮的报警信号或防护区外的紧急启动信号,作为系统的联动触发信号探测器的组合宜采用感烟火灾探测器和感温火灾探测器。
根据《二氧化碳灭火系统设计规范(2010 版) GB50193-93》6.0.2:当采用火灾探测器时 灭火系统的自动控制应在接收到两个独立的火災信号后才能启动。根据人员疏散要求宜延迟启动,但延迟时间不应大于 30s
(2)根据《气体灭火系统施工及验收规范 GB》E.2.3:模拟启动试验結果应符合下列规定:
1)延迟时间与设定时间相符,响应时间满足要求;
2)有关声、光报警信号正确;
3)联动设备动作正确;
4)驱动装置动作可靠
5.分析二氧化碳灭火系统设置有何错误或故障。
(1)机械排风装置设置高度位于房间净高 2/3 以上部位,错误机械排放装置应位于房间下部。
(2)出口处设置有火灾声、光报警器错误。应在入口处设置
(3)气瓶储存压力监测显示为 2.5MPa,压力有故障高压二氧化碳的储存压力取 5.17Mpa
(4)灭火剂输送管道***完毕后,已做水压强度和严密性试验有遗漏。出了做水压试验还需要做气压严密性试验。
(5)灭火设计浓度是灭火浓度的 1.3 倍错误。高压二氧化碳灭火设计浓度应为灭火浓度 1.7 倍且不小于 34%。
【解析】(1)根据《二氧化碳灭火系统设计规范(2010 版) GB50193-93》7.0.5 条文说明:由于二氧化碳比空气重往往聚集在防护区低处,无窗和固定扇的地上防护区以及地下防护区难以采用自然通风的方法将二氧化碳排走因此,应采用机械排风装置并且排风扇的入口应设在防护区的下部。
(2)7.0.1:防护区内应设火灾声报警器必要时,可增设光报警器防护区的人口处应设置火灾声、光报警器。报警时间不宜小于灭火过程所需的时间并应能手动切除报警信号。
(3)二氧化碳灭火系统按灭火剂储存压力不同可分为高压系统(指灭火剂在常温下储存的系统)和低压系统(指灭火剂在-20~-18℃低温下储存的系統)两种应用形式管网起点计算压力(绝对压力):高压系统应取 5.17 MPa,低压系统应取 2.07 MPa
(4)根据《气体灭火系统施工及验收规范 GB》5.5.4:灭火剂输送管道***完毕后,应进行强度试验和气压严密性试验, 并合格
(5)根据《二氧化碳灭火系统设计规范(2010 版) GB50193-93》3.2.1:二氧化碳设计浓度不应小于灭吙浓度的 1.7 倍,并不得低于 34%