COMP9021 Principles of Programming WEEK6

1. Recursion基础

函数不仅可以嵌套,互相调用,而且可以自我调用,这种自我调用可以理解为recursion。为了实现这个目的,recursion的写法有两个重要部分,一个是在函数body中应该出现自我调用,另一个是给出退出和返回机制(base condition),保证recursion可以终止。
recursion的机制是从上到下的寻找base condition,再从下到上的返回每一层recursion,例如:

def f(n):
    if n < 0:
        return
    print(n, end = ' ')
    f(n - 1)
    print(n, end = ' ')

f(4)
>>>
4 3 2 1 0 0 1 2 3 4 

上例说明了recursion的机制,从f(4)进入函数body,进入第二层找f(3),再进入第三层找f(2),继续第四层找f(1),然后来到第五层找f(0),再往下第六层f(-1),触发base condition,于是执行这种情况的代码return,回到第五层f(0)并完成第五层所有代码,这时会再打印一个0,后续依次向上返回,直到完成初始命令f(4)。

2. Recursion练习

写出一段程序,输入一个list,list中的元素都是integer,函数返回这个list所有integer的sum。

def the_sum(L):
    running_sum = 0
    for e in L:
        running_sum += e
    return running_sum

这是iterative的写法,和之前的文章说的方式一致,不多赘述。

def the_sum_recursive_first(L):
    if not L:
        return 0
    return L[0] + the_sum_recursive(L[1: ])

接下来这段代码是recursion的写法。recursion的问题分析方法和iterative 不同,recursion不从case入手分析,而是找base condition和recursion进入下一层的机制。
求一个list的sum,一定是一个一个元素加和起来的,一般情况是从list中找到一个元素,把它和当下的和加总。上面代码是找到list中的第一个元素,所以这个想法的实现代码如下,既完成了求和的目的,又完成自身调用进入了下一层。

return L[0] + the_sum_recursive(L[1: ])

接下来分析base condition,什么时候返回?当list中所有元素都加和过就可以结束,所以实现这个目的的代码如下,不仅结束向下一层的机制,而且要返回求和的base值0。

if not L:
        return 0

既然可以找第一个元素,就可以找最后一个元素,所以同等复杂度的代码如下:

def the_sum_recursive_last(L):
    if not L:
        return 0
    return L[-1] + the_sum_recursive(L[ :-1])

上述的两种recursion的方法层数都很多,效率会低下,一般机制可以继续改善。从一次只加和一个值,到同时加和两个:

def the_sum_two_recursive(L):
    if len(L) == 1:
        return L[0]
    return the_sum_two_recursive(L[: len(L) // 2]) + the_sum_two_recursive(L[len(L) // 2: ])

上述部分Eric的课上代码由冗余,上面写的更加简练。

3. Fibonacci Number

斐波那契数列的每个数等于前两个数字的加和,有初始条件f(0) = 0, f(1) = 1。写成recursion的表达方法:

def fibo(n):
    if n < 2:
        return n
    return fibo(n - 1) + fibo(n - 2)

fibo(5)

接下来会发现效率问题,因为计算f(3)的时候需要f(2)和f(1),而计算f(4)的时候需要一个新的f(3)又要一个计算过的f(2),上述的recursion方法做了大量重复运算。解决方法是记录每一个算出的值,用查询返回值的方法替代重复recursion计算。

def fibo_memorisation(n, results = {0: 0, 1: 1}):
    if n not in results:      
        results[n] = fibo_memorisation(n - 1, results) + fibo_memorisation(n - 2, results)
    return results[n]

fibo_memorisation(40)
--因为函数的第二个input有defalut值,所以调用的时候可以只给出一个input

大部分的recursion都可以用迭代方式解决:

def fibo_iterative(n):
    if n < 2:
        return n
    i = 2
    previous, current = 0, 1
    while i <= n:
        previous, current = current, previous + current
        i += 1
    return current

fibo_iterative(40)

4. Hanoi Question

汉诺塔问题是经典的recursion问题,base条件是一个碟子,general的情况是把其他碟子看做一个整体移动。

def Hanoi_recursive(n, start, spare, finish):
    if n == 1:
        print(f'Move disk from position {start} to position {finish}')
    else:
        Hanoi_recursive(n - 1, start, finish, spare)
        print(f'Move disk from position {start} to position {finish}')
        Hanoi_recursive(n - 1, spare, start, finish)
        
Hanoi_recursive(4, 'A', 'B', 'C')

汉诺塔的iterative的写法比较难懂,Eric给出了PDF解释和相应代码,这里不做讨论。

5. Yield

yield机制非常的python,过去常用的range()就是一个例子。

def f():
    print('Hi')
    yield 0
    print('Hi again')
    yield 1
    print('Hi always')
    yield 2
    print('Bye')
    
F = f()
while True:
    try:
        print(next(F))
    except StopIteration:
        break

>>>
Hi
0
Hi again
1
Hi always
2
Bye

执行函数f()的时候先输出‘Hi',然后给出generate的值0。重复这个过程1和2。最后超出了yield的值的范围,不再输出值。

6. Yield练习

输出所有小于n的质数

from math import sqrt
def primes(n):
    for p in range(2, n + 1):
        for d in range(2, round(sqrt(p)) + 1):
            if p % d == 0:
                break
        else:
            yield p
            
P = primes(100)
while True:
    try:
        print(next(P), end = ' ')
    except StopIteration:
        break

7. Permutation

输入一个数组,给出所有数的全排列:

def permute(L):
    if len(L) <= 1:
        yield L
    #base condition:当只剩下一个元素的时候generate这个元素
    else:
        for i in range(len(L)):
            L1 = list(L)
            #复制list L
            L1[0], L1[i] = L1[i], L1[0]
            #把list中第一个元素和每一个元素都互换,实现全排列
            for e in permute(L1[1: ]):
                yield [L1[0]] + e
                #确定第一个数字后,把剩下的list当成一个新的list,做recursion,yield第一个元素和后面元素的全排列结果
                
P = permute(list(range(4)))
while True:
    try:
        print(next(P))
    except StopIteration:
        break

自己写的排列可能会有使用问题,python自带了排列函数:

from itertools import permutations
P = permutations(list(range(5)))
for p in P:
    print(p)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容