Python装饰器学习过程中遇到一个困扰,请指教

def outer(some_func):
     def inner():
         print ("before some_func")
         ret = some_func()
         return ret
     return inner

def foo(): return 3

outer(foo)()

执行的结果是 before some_func

环境是 python3.6,按思路来说 ret 也应当被打印出来吧?我尝试改成 return print(ret)才能把 3 打印出来。但是不能理解。请各路大佬指点下,谢谢!


Python装饰器学习过程中遇到一个困扰,请指教

8 回复

好糟糕的编辑。。


装饰器这玩意儿说白了就是个语法糖,核心就是函数嵌套和闭包。你遇到的困扰大概率是这几个点:

  1. 装饰器执行时机:装饰器在模块导入时就会执行,不是在函数调用时才执行。比如:
def my_decorator(func):
    print("装饰器执行了!")
    def wrapper():
        return func()
    return wrapper

@my_decorator
def my_func():
    pass  # 这里还没调用函数,装饰器就已经执行了
  1. 函数元信息丢失:用普通嵌套函数做装饰器会导致__name__等属性丢失,得用functools.wraps
from functools import wraps

def decorator(func):
    @wraps(func)  # 这行是关键
    def wrapper(*args, **kwargs):
        return func(*args, **kwargs)
    return wrapper
  1. 带参数的装饰器:这其实是三层嵌套,最外层处理装饰器自己的参数:
def repeat(num):
    def actual_decorator(func):
        @wraps(func)
        def wrapper(*args, **kwargs):
            for _ in range(num):
                result = func(*args, **kwargs)
            return result
        return wrapper
    return actual_decorator
  1. 类装饰器:通过实现__call__方法让类实例能像函数一样调用:
class CountCalls:
    def __init__(self, func):
        self.func = func
        self.calls = 0
    
    def __call__(self, *args, **kwargs):
        self.calls += 1
        return self.func(*args, **kwargs)

建议:多写几个装饰器例子,理解闭包和函数对象的概念。

没 print 怎么打印输出……

是 ide 没自动 echo 吧,你没打印它肯定不出来啊

你按什么思路觉得 ret 应当被打印出来?我不太懂 python,但整个代码只有一句 print (“before some_func”),打印出其他东西才不正常吧?

如果你用 IDLE 的话能看到 3
<br>&gt;&gt;&gt; outer(foo)()<br>before some_func<br>3<br>





多谢各位确实是我理解有问题。用 IDLE 是 OK 的,我傻了 在 pycharm 运行是不打印的。

ret 是被 return 的而不是 print,实际运行中你不会看到 return 但是你手动执行后 return 会显示出来,但实际上并不是 print。

回到顶部