带参数的Python装饰器原来这么简单,5分钟彻底掌握!

带参数的装饰器(三层嵌套结构)

python 复制代码
def repeat_execution(n_times):
    # 第一层:接收装饰器参数
    def decorator(func):
        # 第二层:接收被装饰函数
        def wrapper(*args, **kwargs):
            # 第三层:执行装饰逻辑
            for _ in range(n_times):
                result = func(*args, **kwargs)
            return result
        return wrapper
    return decorator

@repeat_execution(n_times=3)
def say_hello(name):
    print(f"Hello, {name}!")

say_hello("Alice")

输出:

复制代码
Hello, Alice!
Hello, Alice!
Hello, Alice!

深度解析:装饰器参数如何传递

  1. @repeat_execution(n_times=3) 首先执行
    • 调用repeat_execution(3)返回decorator函数
  2. 接着执行@decorator
    • say_hello作为参数传递给decorator
  3. 最终生成装饰后的wrapper函数

带状态记录的装饰器

python 复制代码
def log_performance(metric_name):
    def decorator(func):
        func.call_count = 0  # 添加函数属性
        
        def wrapper(*args, **kwargs):
            func.call_count += 1
            print(f"[{metric_name}] 执行 #{func.call_count}")
            return func(*args, **kwargs)
        return wrapper
    return decorator

@log_performance("用户验证")
def authenticate(user):
    return user == "admin"

print(authenticate("guest"))
print(authenticate("admin"))
print(f"验证函数调用次数: {authenticate.call_count}")

输出:

ini 复制代码
[用户验证] 执行 #1
False
[用户验证] 执行 #2
True
验证函数调用次数: 2

类实现的装饰器参数

python 复制代码
class TimeoutRetry:
    def __init__(self, retries=3, delay=1):
        self.retries = retries
        self.delay = delay
    
    def __call__(self, func):
        import time
        def wrapper(*args, **kwargs):
            for attempt in range(self.retries):
                try:
                    return func(*args, **kwargs)
                except TimeoutError:
                    print(f"超时重试中... ({attempt+1}/{self.retries})")
                    time.sleep(self.delay)
            raise Exception("所有重试均失败")
        return wrapper

@TimeoutRetry(retries=2, delay=0.5)
def fetch_data():
    # 模拟网络请求
    raise TimeoutError("连接超时")

fetch_data()  # 将触发重试机制

装饰器参数的最佳实践

  1. 命名规范 :使用描述性的参数名(如max_retries而非n

  2. 参数验证 :在装饰器最外层验证参数有效性

    python 复制代码
    def validate_params(max_time):
        if max_time <= 0:
            raise ValueError("最大时间必须为正数")
  3. 保留元数据 :使用@functools.wraps保持原始函数属性

    python 复制代码
    from functools import wraps
    
    def debug_decorator(level):
        def decorator(func):
            @wraps(func)  # 保留原始函数信息
            def wrapper(*args, **kwargs):
                if level > 1:
                    print(f"[DEBUG] 调用 {func.__name__}")
                return func(*args, **kwargs)
            return wrapper
        return decorator
相关推荐
麦兜*3 分钟前
Spring Boot 集成Reactive Web 性能优化全栈技术方案,包含底层原理、压测方法论、参数调优
java·前端·spring boot·spring·spring cloud·性能优化·maven
知了一笑13 分钟前
独立开发第二周:构建、执行、规划
java·前端·后端
Otaku love travel27 分钟前
实施运维文档
运维·windows·python
测试老哥1 小时前
软件测试之单元测试
自动化测试·软件测试·python·测试工具·职场和发展·单元测试·测试用例
UI前端开发工作室1 小时前
数字孪生技术为UI前端提供新视角:产品性能的实时模拟与预测
大数据·前端
Sapphire~1 小时前
重学前端004 --- html 表单
前端·html
遇到困难睡大觉哈哈1 小时前
CSS中的Element语法
前端·css
presenttttt1 小时前
用Python和OpenCV从零搭建一个完整的双目视觉系统(六 最终篇)
开发语言·python·opencv·计算机视觉
Real_man1 小时前
新物种与新法则:AI重塑开发与产品未来
前端·后端·面试
小彭努力中1 小时前
147.在 Vue3 中使用 OpenLayers 地图上 ECharts 模拟飞机循环飞行
前端·javascript·vue.js·ecmascript·echarts