递归和循环

当需要重复多次计算相同的问题时,通常可以选择递归或者循环

函数直接或者间接调用自身就是递归

  • 必须有一个明确的递归结束条件,称为递归出口,否则就会无限调用(相当于死循环)。
  • 递归需要有边界条件、递归前进段和递归返回段。
  • 当边界条件不满足时,递归前进。当边界条件满足时,递归返回。

递归函数运行时需要较多次数的函数调用,每次都要创建新的变量,需要增加额外的堆栈处理,会对执行效率有一定影响,占用过多的内存资源。递归有深度限制,如果调用层次过深,函数反复压栈,会造成栈内存溢出。

下面来看两个小例子:

①求n 的阶乘 ,例如5! =5*4*3*2*1

先用循环的方式实现:

def fn(n,s=1):
    for i in range(1,n+1):
         s *=i
    return s

用递归方式实现:

def fn(n):
    if n ==1:           #递归结束条件
        return n
    return n*fn(n-1)   
#fn(5)= 5*fn(4) = 5*4*fn(3) = 5*4*3*fn(2) = 5*4*3*2*fn(1) = 5*4*3*2*1
  
#这是另一种递归形式
def fn(n,s=1):
    if n=1:
        return s
    else:
        return fn(n-1,s*n)     #将累乘的结果传递到下一次调用中去

#fn(5,1)=>fn(4,5)=>fn(3,20)=>fn(2,60)=>fn(1,120)=>120

②猴子第一天摘了若干个桃子,当即吃了一半,还不解馋,又多吃了一个;第二天,吃剩下的桃子的一半,还不过瘾,又多吃了一个;以后每天都吃前一天剩下的一半多一个,到第10天想再吃时,只剩下一个桃子了。问第一天共摘了多少个桃子?

用循环实现:

def peach(day=10,s=1):
    for i in range(day-1):
        s = (s+1)*2
    return s
peach()

#运行结果:1534

用递归求解:

def peach(day=1):
    if day==10:           #设置边界条件
        return 1
    return (peach(day+1)+1)*2

#运行结果:1534
#另一种递归方式求解:

def peach(day=10,s=1):
    if day==1:
        return s
    return peach(day-1,(s+1)*2)   #将上一次计算结果传递到下一次计算中去
peach()

#运行结果:1534

 

递归循环都是计算机程序中常见的控制结构,用于重复执行某个任务。它们在时间复杂度分析中起着关键作用,用来衡量算法执行效率随着输入规模增加而变化的速度。 **1. 递归**: 递归是指一个函数或过程调用自身来解决问题。递归的时间复杂度通常取决于两个因素:基本情况(没有递归调用的情况,其复杂度为 O(1))和递归情况(每次调用的复杂度)。如果递归情况的时间复杂度为 T(n),那么总的时间复杂度将是基本情况加上 n 次递归调用的复杂度,即 T(n) = a * T(n-1) + f(n),其中 a 是递归调用的次数,f(n) 是基本情况的复杂度。如果没有基本情况,或者基本情况的时间复杂度与递归情况相同(例如 Fibonacci 数列),递归的时间复杂度会呈现指数级增长,如 O(2^n) 或者 O(n!)。 **2. 循环**: 循环则直接按照预设的条件重复执行一段代码,比如 while、for 循环循环的时间复杂度通常用循环的次数来表示,假设循环体内的操作复杂度是恒定的,那么循环 n 次的时间复杂度就是 O(n)。如果是嵌套循环,时间复杂度会是两者的乘积,例如外层循环 n 次,内层循环 m 次,总复杂度为 O(n * m)。 **相关问题--:** 1. 递归何时会导致时间复杂度过高? 2. 什么情况下,使用循环递归更优? 3. 在数据结构或算法中,如何选择递归还是循环实现?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值