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 回复
好糟糕的编辑。。
装饰器这玩意儿说白了就是个语法糖,核心就是函数嵌套和闭包。你遇到的困扰大概率是这几个点:
- 装饰器执行时机:装饰器在模块导入时就会执行,不是在函数调用时才执行。比如:
def my_decorator(func):
print("装饰器执行了!")
def wrapper():
return func()
return wrapper
@my_decorator
def my_func():
pass # 这里还没调用函数,装饰器就已经执行了
- 函数元信息丢失:用普通嵌套函数做装饰器会导致
__name__等属性丢失,得用functools.wraps:
from functools import wraps
def decorator(func):
@wraps(func) # 这行是关键
def wrapper(*args, **kwargs):
return func(*args, **kwargs)
return wrapper
- 带参数的装饰器:这其实是三层嵌套,最外层处理装饰器自己的参数:
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
- 类装饰器:通过实现
__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>>>> outer(foo)()<br>before some_func<br>3<br>
多谢各位确实是我理解有问题。用 IDLE 是 OK 的,我傻了 在 pycharm 运行是不打印的。
ret 是被 return 的而不是 print,实际运行中你不会看到 return 但是你手动执行后 return 会显示出来,但实际上并不是 print。

