装饰器-可选参数-《Python教程》

admin 2025-10-19 23:05:42 编程 来源:ZONE.CI 全球网 0 阅读模式

之前我们介绍过了无参数有参数装饰器,那么是否装饰器可以具备可选参数,即既可以使用@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. 参考资料

以太坊cppgolang区别 编程

以太坊cppgolang区别

以太坊是一种去中心化的开源平台,它采用智能合约技术,旨在构建和运行不受干扰的分布式应用程序。作为目前最受欢迎的区块链平台之一,以太坊提供了多种编程语言的支持,其
progolang 编程

progolang

Go语言(Golang)是由Google开发的一门静态类型编程语言。作为一名专业的Golang开发者,我深知这门语言的优势和特点。在本文中,我将介绍Golang
golangn个发送者 编程

golangn个发送者

Golang是一种开源的编程语言,由Google团队开发,旨在提高程序的并发性和简化软件开发过程。在Go语言中,有时需要向多个接收者发送信息。本文将介绍如何在G
golang技能图谱 编程

golang技能图谱

从互联网行业的快速发展到人工智能技术的日益成熟,各种编程语言也应运而生。而在这众多的编程语言中,Golang(即Go)作为一门强大且高效的开发语言备受关注。Go
评论:0   参与:  9