gpt4 book ai didi

python - 使用 functools 的装饰器和不使用 functools 的装饰器有什么区别?

转载 作者:太空宇宙 更新时间:2023-11-03 17:25:42 24 4
gpt4 key购买 nike

我有以下两个装饰器:

装饰器1:

import functools

def overrides(interface_class):
print('interface_class:', interface_class.__name__)

def decorator(method):
print('method:', method.__name__)

@functools.wraps(method)
def func(*args, **kwargs):
assert (method.__name__ in dir(interface_class))
return method

return func

return decorator

装饰器2:

def overrides(interface_class):

def overrider(method):
print('Method name:', method.__name__)
assert(method.__name__ in dir(interface_class))
return method

return overrider

对于装饰器 1,我重写了一个示例,它使用 functools。我在 Stackoverflow 上找到了 Decorator 2 作为一个工作示例。每当我运行包含程序时,装饰器 1 不会执行断言语句,而装饰器 2 则会执行。这两者之间有什么区别,导致了这种情况?

编辑#1:

根据这个问题的答案、第一条评论和另一个例子,我重写了它,现在它可以工作了:

def overrides(interface_class):
def my_decorator(method):
assert (method.__name__ in dir(interface_class)), \
'Trying to override a method named ' + \
method.__name__ + \
' in the interface ' + \
interface_class.__name__ + \
', which does not exist in the interface. Is this a method naming issue?'

@functools.wraps(method)
def wrapped(*args, **kwargs):
return method(*args, **kwargs)

return wrapped

return my_decorator

我将尝试用自己的话解释 assert 语句的位置:因为 assert 语句不是该方法的“新功能”的一部分(事实上​​,没有新功能,只是关于它的断言),它不属于函数 func 内部,而是属于函数外部。它只会在方法被装饰时执行一次,这会在程序启动时发生,因为那是应用函数装饰器的时候,而每次调用装饰函数时都会执行它,如果是的话在那个装饰函数中。由于其性质,每次都运行断言是不可取的。

——这个解释还有错误吗?

最佳答案

当调用方法时,您的第一个装饰器尝试检查该方法是否是接口(interface)的一部分。第二个版本在装饰器运行时(在方法定义时)进行检查。

您的第一个版本中存在错误,这可能会阻止您看到它的工作。最里面的函数正在返回method,而它应该调用它并返回结果:return method(*args, **kwargs)

functools.wraps 函数在第一个装饰器中很有用,因为您要用包装器替换原始方法。 wraps 装饰器使得包装器函数的名称和文档字符串与被包装函数的名称和文档字符串相匹配(以便像 help 这样的东西按预期工作)。第二个装饰器根本不会替换该方法(如果未命中断言,装饰器将原封不动地返回该方法),因此没有包装函数需要 functools.wraps 的帮助。

关于python - 使用 functools 的装饰器和不使用 functools 的装饰器有什么区别?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32597451/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com