包装一个函数隐藏它的属性?
Wrapping a function hides its attributes?
这是我之前问题的延续 。
如果我包装了一个函数,以便它使用以下装饰器记录它被调用的次数:
def keep_count(f):
@wraps(f)
def wrapped_f(*args, **kwargs):
f(*args, **kwargs)
wrapped_f.count += 1
wrapped_f.count = 0
return wrapped_f
然后我想用别的东西再次包装它:
def decorator2(fn):
@wraps(fn)
def fn_wrapper(*args, **kwargs):
if my_condition(fn):
fn(*args, **kwargs)
return fn_wrapper
test_f = decorator2(test_f)
我无法再像我希望的那样访问函数的 count
属性。
count
属性的当前值通过 @wraps(fn)
复制,但如果我再次调用该函数,计数将在原始函数内递增,但新值不会复制到新装饰的功能。
>>> test_f()
() {}
>>> test_f.count
1
>>> test_f = decorator2(test_f)
>>> test_f.count # The 1 gets copied here
1
>>> test_f() # Only the inner function's count increments...
() {}
>>> test_f.count # Still one, tho it should be two
1
有什么解决办法吗?
像 "constantly" 重新包装,或者更好的东西?
functools.wraps()
仅 跨 属性复制。当您随后在包装函数上增加计数器时,您正在为属性分配一个 new 整数值,并且包装器仍将引用旧值。
与其让 wraps
复制属性,不如让它复制函数的整个 __dict__
属性:
from functools import wraps, WRAPPER_ASSIGNMENTS
def decorator2(fn):
@wraps(fn, assigned=WRAPPER_ASSIGNMENTS + ('__dict__',), updated=())
def fn_wrapper(*args, **kwargs):
if my_condition(fn):
fn(*args, **kwargs)
return fn_wrapper
现在包装函数 fn
和 fn_wrapper
对象共享可变 __dict__
命名空间字典,并且对该字典所做的任何更改在两个函数中都是可见的。
assigned
是要复制的属性序列(它通常复制文档字符串、函数名称和模块名称等内容),updated
是应该复制的属性序列像字典一样对待,其中这些字典是从包装函数更新的。后者通常设置为 __dict__
,但现在我们复制了整个对象,我们不再需要从原始对象更新它。
这是我之前问题的延续
如果我包装了一个函数,以便它使用以下装饰器记录它被调用的次数:
def keep_count(f):
@wraps(f)
def wrapped_f(*args, **kwargs):
f(*args, **kwargs)
wrapped_f.count += 1
wrapped_f.count = 0
return wrapped_f
然后我想用别的东西再次包装它:
def decorator2(fn):
@wraps(fn)
def fn_wrapper(*args, **kwargs):
if my_condition(fn):
fn(*args, **kwargs)
return fn_wrapper
test_f = decorator2(test_f)
我无法再像我希望的那样访问函数的 count
属性。
count
属性的当前值通过 @wraps(fn)
复制,但如果我再次调用该函数,计数将在原始函数内递增,但新值不会复制到新装饰的功能。
>>> test_f()
() {}
>>> test_f.count
1
>>> test_f = decorator2(test_f)
>>> test_f.count # The 1 gets copied here
1
>>> test_f() # Only the inner function's count increments...
() {}
>>> test_f.count # Still one, tho it should be two
1
有什么解决办法吗? 像 "constantly" 重新包装,或者更好的东西?
functools.wraps()
仅 跨 属性复制。当您随后在包装函数上增加计数器时,您正在为属性分配一个 new 整数值,并且包装器仍将引用旧值。
与其让 wraps
复制属性,不如让它复制函数的整个 __dict__
属性:
from functools import wraps, WRAPPER_ASSIGNMENTS
def decorator2(fn):
@wraps(fn, assigned=WRAPPER_ASSIGNMENTS + ('__dict__',), updated=())
def fn_wrapper(*args, **kwargs):
if my_condition(fn):
fn(*args, **kwargs)
return fn_wrapper
现在包装函数 fn
和 fn_wrapper
对象共享可变 __dict__
命名空间字典,并且对该字典所做的任何更改在两个函数中都是可见的。
assigned
是要复制的属性序列(它通常复制文档字符串、函数名称和模块名称等内容),updated
是应该复制的属性序列像字典一样对待,其中这些字典是从包装函数更新的。后者通常设置为 __dict__
,但现在我们复制了整个对象,我们不再需要从原始对象更新它。