
在python开发中,装饰器是一种强大且常用的工具,用于在不修改原函数代码的情况下,为其添加额外功能,例如日志记录、权限检查或性能计时。然而,当一个带有计时功能的装饰器被应用于多个函数,并且这些函数之间存在嵌套调用关系时,可能会导致意外的冗余输出。
考虑一个简单的计时装饰器@time_elapsed,它记录并打印函数的执行时间:
import time
from functools import wraps
def time_elapsed(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        start_time = time.time()
        result = func(*args, **kwargs)
        elapsed_time = time.time() - start_time
        print(f'{func.__name__} took {elapsed_time:.2f} seconds.')
        return result
    return wrapper
@time_elapsed
def func1():
    time.sleep(0.1)
@time_elapsed
def func2():
    func1() # func1 is called within func2
    time.sleep(0.2)当独立调用func1()时,输出符合预期:
func1 took 0.10 seconds.
然而,当调用func2()时,由于func1在func2内部被调用,func1的计时信息也会被打印出来,导致如下的冗余输出:
func1 took 0.10 seconds. func2 took 0.30 seconds.
这与我们通常希望只看到最外层函数func2的计时结果的期望不符。为了解决这个问题,我们需要一种机制来区分当前函数调用是独立的主调用,还是某个外层函数内部的嵌套调用。
立即学习“Python免费学习笔记(深入)”;
解决此问题的核心思想是在装饰器内部维护一个全局或装饰器级别的调用深度计数器。当一个被装饰的函数被调用时,我们首先检查当前的调用深度。如果深度超过预设的阈值,则跳过计时和打印;否则,执行计时逻辑并递增计数器,在函数执行完毕后递减计数器。
以下是修改后的time_elapsed装饰器实现:
import time
from functools import wraps
def time_elapsed(func):
    # 定义计时打印的深度阈值。
    # DEPTH = 1 意味着只打印最外层被装饰函数的计时。
    # DEPTH = 2 意味着打印最外层及其直接子函数的计时,以此类推。
    DEPTH = 1
    # 初始化一个装饰器级别的计数器。
    # 使用函数属性来存储状态,确保每次调用time_elapsed装饰器时,
    # 都能访问到同一个计数器实例。
    if not hasattr(time_elapsed, '_timer_running'):
        time_elapsed._timer_running = 0
    @wraps(func)
    def wrapper(*args, **kwargs):
        # 如果当前调用深度已达到或超过设定的阈值,
        # 则直接执行原函数,不进行计时和打印。
        if time_elapsed._timer_running >= DEPTH:
            return func(*args, **kwargs)
        # 否则,递增计数器,表示进入了一个新的计时层级。
        time_elapsed._timer_running += 1
        # 执行计时逻辑
        start_time = time.time()
        result = func(*args, **kwargs)
        elapsed_time = time.time() - start_time
        print(f'{func.__name__} took {elapsed_time:.2f} seconds.')
        # 函数执行完毕后,递减计数器,退出当前计时层级。
        time_elapsed._timer_running -= 1
        return result
    return wrapper让我们使用这个改进后的装饰器来测试多层嵌套的函数调用:
@time_elapsed
def func1():
    time.sleep(0.1)
@time_elapsed
def func2():
    func1()
    time.sleep(0.2)
@time_elapsed
def func3():
    func1()
    func2()
    time.sleep(0.3)
@time_elapsed
def func4():
    func1()
    func2()
    func3()
    time.sleep(0.4)
if __name__ == "__main__":
    print("--- func1 ---")
    func1()
    print("\n--- func2 ---")
    func2()
    print("\n--- func3 ---")
    func3()
    print("\n--- func4 ---")
    func4()当DEPTH设置为1时,只有最外层的函数调用会打印计时信息:
--- func1 --- func1 took 0.10 seconds. --- func2 --- func2 took 0.30 seconds. --- func3 --- func3 took 0.70 seconds. --- func4 --- func4 took 1.50 seconds.
可以看到,func2调用时不再打印func1的计时,func3调用时不再打印func1和func2的计时,以此类推。这正是我们期望的“只打印最外层”行为。
如果我们修改time_elapsed装饰器中的DEPTH为2:
def time_elapsed(func):
    DEPTH = 2 # 允许打印两层深度的计时
    # ... (其余代码不变)再次运行上述测试代码,输出将变为:
--- func1 --- func1 took 0.10 seconds. --- func2 --- func1 took 0.10 seconds. func2 took 0.30 seconds. --- func3 --- func1 took 0.10 seconds. func2 took 0.30 seconds. func3 took 0.70 seconds. --- func4 --- func1 took 0.10 seconds. func2 took 0.30 seconds. func3 took 0.70 seconds. func4 took 1.50 seconds.
此时,func2调用时会打印func1的计时,因为它处于第一层嵌套(深度为2)。func3调用时,func1和func2的计时也会被打印,因为它们都在允许的深度范围内。但如果func2内部再调用一个被装饰的函数,且该函数是func3的第三层嵌套,则其计时将不会被打印。
通过引入调用深度计数器,我们成功地将一个普通的计时装饰器升级为智能型,能够有效避免嵌套函数调用中的冗余输出,提供更精确和可控的性能监控体验。
以上就是Python装饰器在嵌套函数调用中避免重复计时输出的策略的详细内容,更多请关注php中文网其它相关文章!
                        
                        每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
                Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号