Retry装饰器, 类装饰器

这里说类装饰器,就不得不说一个特殊的函数__call__

在Python中,函数其实就是一个对象:

>>> f = abs
>>> f.__name__
'abs'
>>> f(-123)
123

由于f可以被调用,所以f被称为可调用对象

所有的函数都是可调用对象

一个类实例也可以变成一个可调用对象,只需要实现一个特殊的方法__call__()

我们把Person类变成一个可调用对象

class Person(object):
    def __init__(self, name, gender):
        self.name = name
        self.gender = gender

    def __call__(self, friend):
        print 'My name is %s...' % self.name
        print 'My friend is %s...' % friend

现在可以对Person实例直接调用

>>> p = Person('Bob', 'male')
>>> p('Tim')
My name is Bob...
My friend is Tim...

这样的话,也就是说我们可以把类写成装饰器

class Deco:    
    def __init__(self, func):       
        self.func = func

   def __call__(self, *args, **kwargs):        
        return self.func(*args, **kwargs)
@Deco 
def foo(x, y):    
    return x ** y
# 过程解析
fn = Deco(foo) 
foo = fn 
foo(12, 34)

下面写一个在工作中常用的重试装饰器的简单实现Retry

import time
class retry(object):   
    def __init__(self, max_retries=3, wait=0, exceptions= (Exception,)):        
        self.max_retries = max_retries        
        self.exceptions = exceptions        
        self.wait = wait

    def __call__(self, func):        
        def wrapper(*args, **kwargs):
             for i in range(self.max_retries + 1):                
                 try:                   
                    result = func(*args, **kwargs)                
                 except self.exceptions:                    
                    time.sleep(self.wait)                    
                    continue                
                 else:                    
                    return result        
        return wrapper




在retry装饰器函数中,可以通过设置`retry_on_exception`参数来指定哪些异常需要重试,同时可以通过设置`wrap_exception`参数来控制是否在重试时打印出异常信息。 如果希望在每次重试时都打印出异常信息,可以设置`wrap_exception`参数为True。示例代码如下: ```python import time from retry import retry @retry(delay=1, backoff=2, max_delay=4, retry_on_exception=lambda e: isinstance(e, ValueError), wrap_exception=True) def my_func(): print("Running my_func...") if time.time() % 2 == 0: raise ValueError("Oops! Something went wrong...") else: print("Success!") my_func() ``` 在这个示例代码中,我们设置了`wrap_exception`参数为True,表示在每次重试时打印出异常信息。当函数第一次运行时,它会输出“Running my_func...”和“Success!”,并正常结束。当函数第二次运行时,它会抛出一个ValueError异常,并在控制台上打印出异常信息,然后重试多次,直到成功为止。 如果不想在每次重试时打印出异常信息,可以将`wrap_exception`参数设置为False。如下所示: ```python import time from retry import retry @retry(delay=1, backoff=2, max_delay=4, retry_on_exception=lambda e: isinstance(e, ValueError), wrap_exception=False) def my_func(): print("Running my_func...") if time.time() % 2 == 0: raise ValueError("Oops! Something went wrong...") else: print("Success!") my_func() ``` 在这个示例代码中,我们设置了`wrap_exception`参数为False,表示在每次重试时不打印出异常信息。当函数第二次运行时,它会抛出一个ValueError异常,并重试多次,直到成功为止,而不会在控制台上打印出异常信息。 希望这个示例能够帮助你解决问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值