在Python中使用generator将递归转换为迭代

在Python中使用generator将递归转换为迭代

trampoline.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
def trampoline(f, stack=[]):
def wrapped(*args, **kwargs):
if stack:
return f(*args, **kwargs)
stack.append(f(*args, **kwargs))
val = None
while stack:
try:
stack.append(stack[-1].send(val))
val = None
except StopIteration as e:
stack.pop()
val = e.value
return val

return wrapped


if __name__ == "__main__":

@trampoline
def fib(x: int):
if x <= 1:
return 1
return (yield fib(x - 1)) + (yield fib(x - 2))

@trampoline
def is_odd(x: int):
if x == 0:
return False
return (yield is_even(x - 1))

@trampoline
def is_even(x: int):
if x == 0:
return True
return (yield is_odd(x - 1))

@trampoline
def factorial(x: int):
if x == 0:
return 1
return (yield factorial(x - 1)) * x % (int(1e9) + 7)

print(fib(10))
print(is_odd(100000))
print(is_even(100000))
print(factorial(100000))
PYTHON

以及pajenegod另一种写法

bootstrap.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
from types import GeneratorType


def bootstrap(f, stack=[]):
def wrappedfunc(*args, **kwargs):
if stack:
return f(*args, **kwargs)
else:
to = f(*args, **kwargs)
while True:
if isinstance(to, GeneratorType):
stack.append(to)
to = next(to)
else:
stack.pop()
if not stack:
break
to = stack[-1].send(to)
return to

return wrappedfunc


if __name__ == "__main__":

@bootstrap
def fib(x: int):
if x <= 1:
yield 1
yield (yield fib(x - 1)) + (yield fib(x - 2))

@bootstrap
def is_odd(x: int):
if x == 0:
yield False
yield (yield is_even(x - 1))

@bootstrap
def is_even(x: int):
if x == 0:
yield True
yield (yield is_odd(x - 1))

@bootstrap
def factorial(x: int):
if x == 0:
yield 1
yield (yield factorial(x - 1)) * x % (int(1e9) + 7)

print(fib(10))
print(is_odd(100000))
print(is_even(100000))
print(factorial(100000))
PYTHON

在Python中使用generator将递归转换为迭代
https://blog.fredbill.eu.org/programming/python-convert-recursion-to-iteration-using-generator/
作者
FredBill
发布于
2025年2月8日
许可协议