关于递归深度受限和堆栈溢出4种解决办法

# -*- coding: utf-8 -*-
"""
@Time    : 2023/1/19  019 下午 16:31
@Author  : Jan
@File    : 递归深度和堆栈溢出.py
"""
import sys
import types

""" {如下共四种解决办法} """


class TailRecurseException(BaseException):
    def __init__(self, args, kwargs):
        self.args = args
        self.kwargs = kwargs


def tail_call_optimized(g):
    """
    This function decorates a function with tail call
    optimization. It does this by throwing an exception
    if it is it's own grandparent, and catching such
    exceptions to fake the tail call optimization.

    This function fails if the decorated5
    function recurses in a non-tail context.
    """

    def func(*args, **kwargs):
        f = sys._getframe()
        if f.f_back and f.f_back.f_back and f.f_back.f_back.f_code == f.f_code:
            raise TailRecurseException(args, kwargs)
        else:
            while 1:
                try:
                    return g(*args, **kwargs)
                except TailRecurseException as e:
                    args = e.args
                    kwargs = e.kwargs

    func.__doc__ = g.__doc__
    return func


# 方法一:
# 尾递归优化,默认不带装饰器会超系统默认递归深度 1000
# sys.setrecursionlimit(3000)  # 可以修改默认递归深度
# threading.stack_size(12000000)  # 修改堆栈大小时需要单开进程,并且要结合上面递归深度一起设置才有效
@tail_call_optimized  # 使用时有个条件是返回递归函数外部不能带有计算表达式
def f0(i, res=0):
    if i == 0:
        return res + i
    return f0(i - 1, res + i)


result = f0(9999)
print(result, '----f0')


# 方法二:
# 用 while 循环代替
def f1(i, res=0):
    while i >= 0:
        res += i
        i -= 1
    return res


result = f1(9999)
print(result, '----f1')


# 方法三:
# 使用 “尾递归 + 生成器” 彻底解决堆栈溢出
def f_recursive(i, res=1):
    if i == 1:
        yield res
    yield f_recursive(i - 1, res + i)


def f_recursive_wapper(generator, i):
    gen = generator(i)
    while isinstance(gen, types.GeneratorType):
        gen = gen.__next__()
    return gen


result = f_recursive_wapper(f_recursive, 9999)
print(result, '----f_recursive')


# 应用测试效果
def advanced(a, b, c, d, k=0):  # k 是自增变量(如下 m-n 的值),可取值 0 - n ,直到满足条件
    # 原理是假设各自商为 m,n ,对各自乘以对方的除数,得到新除数为二者最小公倍数,故得知 acm+bc=cN,acn+ad=aN ,再带入其求差计算出 N 的通式
    x, y = divmod(a * c * k + b * c - a * d, c - a)
    if y == 0 and x > 0:
        yield x
    else:
        yield advanced(a, b, c, d, k + 1)


def f_recursive_wapper2(generator, a, b, c, d):
    gen = generator(a, b, c, d)
    while isinstance(gen, types.GeneratorType):
        gen = gen.__next__()
    return gen


a, b = [5, 4]
c, d = [7, 6]
e, f = [9, 8]
g, h = [11, 0]
x, y = a * c, f_recursive_wapper2(advanced, a, b, c, d)
x, y = e * x, f_recursive_wapper2(advanced, e, f, x, y)
x, y = g * x, f_recursive_wapper2(advanced, g, h, x, y)
N1 = f_recursive_wapper2(advanced, 1, 0, x, y)
for n in range(20):
    N = 1 * x * n + N1
    print(N, end=',')
print('----------4.1.1')

全部评论

相关推荐

#简历#先说一说我自己的想法,很多人都很排斥苍穹外卖,认为没什么技术点和含金量,但实际上我觉得恰恰相反,苍穹外卖虽然代码本身并不是你自身能力的证明,但是是作为一个新人学习时很好的跳板和原始框架,在这个框架上进行的改进可以很好的辐射到你自己的个人成果上,并作为你和面试官聊天的筹码大多数人的苍穹外卖只写增删改查,千篇一律,吸引不了面试官,所以这才让大家误以为只要是苍穹外卖就不要写进简历里这种误区,但实际上如果你在原有的层面上进行改进,并作为你的项目亮点和面试官介绍,告诉他你的苍穹外卖和别人的有什么不同,增加了哪些技术难点,这才显得你是完全自己理解了这个项目,并且有自己动手实践项目的能力,而不是就看了个课程就以为自己会了,就当成自己的了,如此一来,这反而成为你的加分项苍穹外卖为什么看的人最多,说明它好啊,如果它不好,为什么看的人还这么多,想清楚这个逻辑,我觉得要做的最重要的事,就是如何在原有框架上进行改进提效,比起听其他人的话重新搞一个项目性价比高得多,而且我亲测项目并没有成为我找到工作的阻碍,我投的大厂一大半都给我面试了,而且很多不止一个部门,退一万步说,当你手头没有其他项目的时候,有苍穹外卖总比什么都没有的好很多,不需要因为苍穹外卖有任何心理负担关于简历的任何部分都欢迎大家提意见,十分感谢大家,祝大家找实习+秋招顺利上岸,offer拿到手软#简历中的项目经历要怎么写##我的上岸简历长这样##最后再改一次简历##简历##简历被挂麻了,求建议#
点赞 评论 收藏
转发
点赞 收藏 评论
分享
牛客网
牛客企业服务