python二分算法递归_Python递归函数,二分查找算法

1、初始递归

递归函数:在一个函数里在调用这个函数自己。html

递归的最大深度:998python

正如大家刚刚看到的,递归函数若是不受到外力的阻止会一直执行下去。可是咱们以前已经说过关于函数调用的问题,每一次函数调用都会产生一个属于它本身的名称空间,若是一直调用下去,就会形成名称空间占用太多内存的问题,因而python为了杜绝此类现象,强制的将递归层数控制在了997(只要997!你买不了吃亏,买不了上当...).算法

拿什么来证实这个“998理论”呢?这里咱们能够作一个实验:函数

def foo(n):

print(n)

n += 1

foo(n)

foo(1)

由此咱们能够看出,未报错以前能看到的最大数字就是998.固然了,997是python为了咱们程序的内存优化所设定的一个默认值,咱们固然还能够经过一些手段去修改它:性能

import sys

print(sys.setrecursionlimit(100000))

咱们能够经过这种方式来修改递归的最大深度,刚刚咱们将python容许的递归深度设置为了10w,至于实际能够达到的深度就取决于计算机的性能了。不过咱们仍是不推荐修改这个默认的递归深度,由于若是用997层递归都没有解决的问题要么是不适合使用递归来解决要么是你代码写的太烂了~~~学习

看到这里,你可能会以为递归也并非多么好的东西,不如while True好用呢!然而,江湖上流传这这样一句话叫作:人理解循环,神理解递归。因此你可别小看了递归函数,不少人被拦在大神的门槛外这么多年,就是由于没能领悟递归的真谛。并且以后咱们学习的不少算法都会和递归有关系。来吧,只有学会了才有资本嫌弃!优化

2、递归示例讲解

这里咱们又要举个例子来讲明递归能作的事情。spa

例一:code

如今大家问我,alex老师多大了?我说我不告诉你,但alex比 egon 大两岁。htm

你想知道alex多大,你是否是还得去问egon?egon说,我也不告诉你,但我比武sir大两岁。

你又问武sir,武sir也不告诉你,他说他比太白大两岁。

那你问太白,太白告诉你,他18了。

这个时候你是否是就知道了?alex多大?

1

金鑫

18

2

武sir

20

3

egon

22

4

alex

24

你为何能知道的?

首先,你是否是问alex的年龄,结果又找到egon、武sir、太白,你挨个儿问过去,一直到拿到一个确切的答案,而后顺着这条线再找回来,才获得最终alex的年龄。这个过程已经很是接近递归的思想。咱们就来具体的我分析一下,这几我的之间的规律。

age(4) = age(3) + 2

age(3) = age(2) + 2

age(2) = age(1) + 2

age(1) = 40

那这样的状况,咱们的函数怎么写呢?

def age(n):

if n == 1:

return 40

else:

return age(n-1)+2

print(age(4))

若是有这样一个列表,让你从这个列表中找到66的位置,你要怎么作?

l = [2,3,5,10,15,16,18,22,26,30,32,35,41,42,43,55,56,66,67,69,72,76,82,83,88]

你说,so easy!

l.index(66)...

咱们之因此用index方法能够找到,是由于python帮咱们实现了查找方法。若是,index方法不给你用了。。。你还能找到这个66么?

l = [2,3,5,10,15,16,18,22,26,30,32,35,41,42,43,55,56,66,67,69,72,76,82,83,88]

i = 0

for num in l:

if num == 66:

print(i)

i+=1

上面这个方法就实现了从一个列表中找到66所在的位置了。

但咱们如今是怎么找到这个数的呀?是否是循环这个列表,一个一个的找的呀?假如咱们这个列表特别长,里面好好几十万个数,那咱们找一个数若是运气很差的话是否是要对比十几万次?这样效率过低了,咱们得想一个新办法。

二分查找算法

l = [2,3,5,10,15,16,18,22,26,30,32,35,41,42,43,55,56,66,67,69,72,76,82,83,88]

你观察这个列表,这是否是一个从小到大排序的有序列表呀?

若是这样,假如我要找的数比列表中间的数还大,是否是我直接在列表的后半边找就好了?

这就是二分查找算法!

那么落实到代码上咱们应该怎么实现呢?

简单版二分法

l = [2,3,5,10,15,16,18,22,26,30,32,35,41,42,43,55,56,66,67,69,72,76,82,83,88]

def func(l,aim):

mid = (len(l)-1)//2

if l:

if aim > l[mid]:

func(l[mid+1:],aim)

elif aim < l[mid]:

func(l[:mid],aim)

elif aim == l[mid]:

print("bingo",mid)

else:

print('找不到')

func(l,66)

func(l,6)

升级版二分法

l1 = [1, 2, 4, 5, 7, 9]

def two_search(l,aim,start=0,end=None):

end = len(l)-1 if end is None else end

mid_index = (end - start) // 2 + start

if end >= start:

if aim > l[mid_index]:

return two_search(l,aim,start=mid_index+1,end=end)

elif aim < l[mid_index]:

return two_search(l,aim,start=start,end=mid_index-1)

elif aim == l[mid_index]:

return mid_index

else:

return '没有此值'

else:

return '没有此值'

print(two_search(l1,9))

原文出处:https://www.cnblogs.com/changxin7/p/11340202.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值