python在expect异常之后返回try块

2024-04-27 03:05:00 发布

您现在位置:Python中文网/ 问答频道 /正文

我跟踪了一些递归函数的装饰器。我想知道怎么回到try街区。我尝试过while循环,但它对我不起作用,因为我的函数是递归的,有人能告诉我如何处理它吗? 问题是当函数改变时,我想继续执行我的 try块

这是我的装饰师和功能

正确结果:

,- change_t([9, 7, 5], 44)
| ,- change_t([9, 7, 5], 35)
| | ,- change_t([9, 7, 5], 26)
| | | ,- change_t([9, 7, 5], 17)
| | | | ,- change_t([9, 7, 5], 8)
| | | | | ,- change_t([7, 5], 8)
| | | | | | ,- change_t([7, 5], 1)
| | | | | | | ,- change_t([5], 1)
| | | | | | | | ,- change_t([], 1)
| | | | | | ,- change_t([5], 8)
| | | | | | | ,- change_t([5], 3)
| | | | | | | | ,- change_t([], 3)
| | | | | | | ,- change_t([], 8)
| | | | ,- change_t([7, 5], 17)
| | | | | ,- change_t([7, 5], 10)
| | | | | | ,- change_t([7, 5], 3)
| | | | | | | ,- change_t([5], 3)
| | | | | | | | ,- change_t([], 3)
| | | | | | ,- change_t([5], 10)
| | | | | | | ,- change_t([5], 5)
| | | | | | | | ,- change_t([5], 0)
| | | | | | | | `- []
| | | | | | | `- [5]
| | | | | | `- [5, 5]
| | | | | `- [5, 5]
| | | | `- [7, 5, 5]
| | | `- [7, 5, 5]
| | `- [9, 7, 5, 5]
| `- [9, 9, 7, 5, 5]
`- [9, 9, 9, 7, 5, 5]

我得到的是: 在我预料到一个例外之后就停止了

^{pr2}$

Tags: 函数功能装饰changetrywhile装饰师继续执行
2条回答

当引发异常时,将从引发异常的代码中放弃控制。而是将它交给第一个捕捉异常的catch块,或者交给主解释器循环。在

看看你的代码:

if a==0:
    return []
elif len(l)==0:
    raise ChangeException()
elif l[0]>a:
    return change_t(l[1:],a)

当第三个条件为真时,即l[0]>a,当l的长度为1时会发生什么?然后在下一个调用中,即return change_t(l[1:],a)它将引发一个异常,除了主循环之外,任何人都不会捕捉到该异常:这就是代码失败的原因。您需要将第三个条件包装在try-catch子句中,这取决于您想要实现的目标。在

正如我在评论中所说的,您需要重新定义异常,以便原始函数能够实际捕获并继续。您希望在except处理程序中只将缩进减少一个,这样就不会在左侧结束得太远:

class traced(object):
    indent =0
    def __init__(self,f):
        self.__name__=f.__name__
        self.indent=0
        self.f=f         
    def __call__(self,*args,**kwargs):
        string=""           
        if kwargs:
           l=[]
           for (key, value) in kwargs.items():
               l.append(str(key) + "=" + str(value))
           a=', '.join(l)
           string = '('+a+')'              
        else:
             l=[]
             for value in args:
                 l.append(str(value))
             a=', '.join(l)
             string = '('+a+')'       
        print('| ' * traced.indent + ',- '+ self.__name__+' '+string)   
        try:
            traced.indent+=1
            value = self.f(*args,**kwargs)                
        except Exception:
            traced.indent-=1  # <  only decrement by one
            raise             # <  reraise the exception so the original function can catch it
        traced.indent-=1
        print('| '* traced.indent + "`- "+ repr(value))          
        return value

然后它就起作用了:

^{pr2}$

最后,我会清理一下装修工,让它更简洁明了,你在做什么:

class traced(object):
    indent = 0

    def __init__(self, f):
        self.__name__ = f.__name__
        self.f = f

    def __call__(self, *args, **kwargs):
        if kwargs:
            l = [str(key) + '=' + str(value) for key, value in kwargs.items()]
        else:
            l = list(map(str, args))
        print('| ' * traced.indent + ',- {0} ({1})'.format(self.__name__, ', '.join(l)))
        try:
            traced.indent += 1
            value = self.f(*args,**kwargs)                
        finally:
            traced.indent -= 1

        print('| ' * traced.indent + '`- ' + repr(value))
        return value

在这里,我简化了整个参数聚合以使用列表理解。我还使用了字符串格式,使格式更加清晰。这样,您还可以组合必须放在列表内容周围的括号(在这两种情况下都是这样做的)。当你在没有真正查看异常的情况下重读异常时,我们不需要从一开始就捕捉它,只需确保我们调整了finally块中的缩进。在

实际上,为什么要检查变量参数关键字参数?接受这两者:

l = list(map(str, args))
l.extend([str(key) + '=' + str(value) for key, value in kwargs.items()])

相关问题 更多 >