之前我们介绍过了无参数有参数装饰器,那么是否装饰器可以具备可选参数,即既可以使用@func,也可以使用@func(a=1, b=2),答案是可以的,本篇文章我们介绍一下如何构建一个可选参数的装饰器。

我们所实现的装饰器是一个任务重启装饰器,即任务失败时,能根据指定次数和间隔时间重新运行函数。

1. 借助args

第一种方式是借助于args参数,即通过判断其长度与值的类型,在内部重新定义retry_timessleep_time变量:

  1. import time
  2. import logging
  3. from functools import wraps
  4. def retry(*args):
  5. def func_retry(func):
  6. log = logging.getLogger("Retry")
  7. @wraps(func)
  8. def wrapper(*args, **kwargs):
  9. cnt = 1
  10. while cnt <= retry_times:
  11. try:
  12. res = func(*args, **kwargs)
  13. return res
  14. except Exception as e:
  15. log.error(f"Retry [red]{cnt}[/] times \n", extra={"markup": True})
  16. cnt += 1
  17. time.sleep(sleep_time)
  18. log.error("Messages: [red]{0}[/]".format(e), extra={"markup": True})
  19. else:
  20. log.exception(f"Retry {retry_times} times, but all failed!")
  21. raise Exception(f"Retry {retry_times} times, but all failed!")
  22. return wrapper
  23. log = logging.getLogger("Retry")
  24. # 对应@retry
  25. if len(args) == 1 and callable(args[0]):
  26. retry_times, sleep_time = 5, 60
  27. return func_retry(args[0])
  28. # 对应@retry(3)
  29. elif len(args) == 1 and isinstance(args[0], int):
  30. retry_times, sleep_time = args[0], 60
  31. return func_retry
  32. # 对应@retry(2, 3)
  33. else:
  34. retry_times, sleep_time = args[0], args[1]
  35. return func_retry

按照上述方式所实现的装饰器,使用方法如下:

  1. @retry
  2. def retry_func():
  3. print(1 / 0)
  4. @retry(2)
  5. def retry_func():
  6. print(1 / 0)
  7. @retry(2, 3)
  8. def retry_func():
  9. print(1 / 0)

可以看到,这种实现方式并不好,它并没有采用传参的方式,并且使用者对所传入值代表含义也会有疑惑。

2. 更简洁的实现方式

接下来我们介绍一种更好的实现方式,借助于functools模块中的partial函数:

  1. import time
  2. import logging
  3. from functools import wraps, partial
  4. def retry(func = None, *, retry_times: int = 5, sleep_time: int = 60):
  5. log = logging.getLogger("Retry")
  6. if func is None:
  7. return partial(retry, retry_times=retry_times, sleep_time=sleep_time)
  8. @wraps(func)
  9. def wrapper(*args, **kwargs):
  10. cnt = 1
  11. while cnt <= retry_times:
  12. try:
  13. res = func(*args, **kwargs)
  14. return res
  15. except Exception as e:
  16. log.error(f"Retry [red]{cnt}[/] times \n", extra={"markup": True})
  17. cnt += 1
  18. time.sleep(sleep_time)
  19. log.error("Messages: [red]{0}[/]".format(e), extra={"markup": True})
  20. else:
  21. log.exception(f"Retry {retry_times} times, but all failed!")
  22. raise Exception(f"Retry {retry_times} times, but all failed!")
  23. return wrapper

显然,这种方式实现的装饰器代码简洁多了,具体使用方法如下:

  1. @retry
  2. def retry_func():
  3. print(1 / 0)
  4. @retry(retry_times=2)
  5. def retry_func():
  6. print(1 / 0)
  7. @retry(retry_times=2, sleep_time=3)
  8. def retry_func():
  9. print(1 / 0)

3. 参考资料