[英]Python 3 - Decorators execution flow
以下示例摘自python Cookbook第三版9.5节。 我在每一行都放置了断点,以了解执行流程。 以下是代码示例,其输出和我所遇到的问题。 我试图解释我的问题,如果您需要更多信息,请告诉我。
from functools import wraps, partial
import logging
# Utility decorator to attach a function as an attribute of obj
def attach_wrapper(obj, func=None):
if func is None:
return partial(attach_wrapper, obj)
setattr(obj, func.__name__, func)
return func
def logged(level, name=None, message=None):
def decorate(func):
logname = name if name else func.__module__
log = logging.getLogger(logname)
logmsg = message if message else func.__name__
@wraps(func)
def wrapper(*args, **kwargs):
log.log(level, logmsg)
return func(*args, **kwargs)
@attach_wrapper(wrapper)
def set_message(newmsg):
nonlocal logmsg
logmsg = newmsg
return wrapper
return decorate
# Example use
@logged(logging.DEBUG)
def add(x, y):
return x + y
logging.basicConfig(level=logging.DEBUG)
add.set_message('Add called')
#add.set_level(logging.WARNING)
print (add(2, 3))
输出是
DEBUG:__main__:Add called
5
我了解装饰器的概念,但这有点令人困惑。
方案1 。 当调试以下行@logged(logging.DEBUG)时 ,我们在0x000000000 <memoryaddress >>下获得decorate = .decorate。
问题 :为什么控件会返回执行“ def decorate”功能? 是因为“装饰”功能位于堆栈的顶部吗?
方案2 :执行@attach_wrapper(wrapper)时 ,控件转到执行attach_wrapper(obj,func = None) ,部分函数返回func =
问题 :为什么控件将返回执行def attach_wrapper(obj,func = None):这次将如何将func的值为* .decorate..set_message在0x000000000>传递给attach_wrapper?
这个:
@logged(logging.DEBUG)
def add(x, y):
....
与此相同:
def add(x, y):
....
add = logged(logging.DEBUG)(add)
请注意,这里有两个调用:首先logged(logging.DEBUG)
返回decorate
,然后调用decorate(add)
。
与方案1中的相同:
@attach_wrapper(wrapper)
def set_message(newmsg):
...
与此相同:
def set_message(newmsg):
...
set_message = attach_wrapper(wrapper)(set_message)
同样,有两个调用:首先attach_wrapper(wrapper)
返回partial
对象,然后调用partial(set_message)
。
logged
和attach_wrapper
不是装饰器。 这些是返回装饰器的函数。 这就是为什么要进行两次调用:一个调用返回装饰器的函数,另一个调用装饰器本身。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.