gpt4 book ai didi

python - 将装饰器类应用于类方法

转载 作者:行者123 更新时间:2023-11-28 17:28:39 25 4
gpt4 key购买 nike

考虑以下装饰器:

class connector(object):
def __init__(self, signal):
self.signal = signal
def __call__(self, slot_func):
def wrapper(*args, **kwargs):
slot_func(*args, **kwargs)
self.signal.connect(wrapper)

然后是我需要修饰的信号和方法类:

from signalslot import Signal

update = Signal()

class manager(object):
# SOME CODE CUT
@connector(update)
def update(self):
print("I'm updating, yay!!!!")

如您所见,我需要向装饰器传递一些额外的参数,在这种情况下 - 我需要连接的信号。如何也通过 self ?

我之所以这么问,是因为如果我尝试将此装饰器应用于方法而不是函数,它会失败并出现以下错误:

TypeError: update() missing 1 required positional argument: 'self'

更具体地说,如果我尝试发出信号:

update.emit()

是的,我使用 "signalslot"在那个项目中。

最佳答案

self 必须是位置参数,而不是关键字参数:

def wrapper(self, *args, **kwargs):
# ^^^^ You can't use "self=None" here.
slot_func(self, *args, **kwargs)

如果你需要区分函数和方法,实现一个descriptor相反。

但是,如果您尝试连接信号,则需要每个实例上的绑定(bind)方法。您最好在实例创建时连接您的信号:

class manager(object):
def __init__(self):
update.connect(self.update)

def update(self):
print("I'm updating, yay!!!!")

manager.__init__ 被调用时,你有一个新的实例,然后你可以创建 one self.update 绑定(bind)方法接收信号。

您仍然可以为此使用装饰器,但是您最多可以在类级别注册哪些函数可以充当信号处理程序;您必须在创建实例时枚举类中的所有函数,然后绑定(bind)所有这些信号:

class connector(object):
def __init__(self, signal):
self.signal = signal
def __call__(self, slot_func):
slot_func._signal_handler = self.signal
return slot_func

和一个单独的类装饰器来包装 class.__init__ 方法:

from inspect import getmembers, isfunction

def connectsignals(cls):
signal_handlers = getmembers(
cls, lambda m: isfunction(m) and hasattr(m, '_signal_handler'))
init = getattr(cls, '__init__', lambda self: None)
def wrapper(self, *args, **kwargs):
init(self, *args, **kwargs)
for name, handler in signal_handlers:
handler._signal_handler.connect(handler.__get__(self))
cls.__init__ = wrapper
return cls

装饰类和信号处理程序:

@connectsignals
class manager(object):
@connector(update)
def update(self):
print("I'm updating, yay!!!!")

装饰器然后在每次创建新实例时连接所有处理程序:

>>> class Signal(object):
... def connect(self, handler):
... print('connecting {!r}'.format(handler))
...
>>> update = Signal()
>>> @connectsignals
... class manager(object):
... @connector(update)
... def update(self):
... print("I'm updating, yay!!!!")
...
>>> manager()
connecting <bound method manager.update of <__main__.manager object at 0x105439ac8>>
<__main__.manager object at 0x105439ac8>

您可能想检查 signalslot 项目是否使用弱引用来跟踪信号处理程序,因为您可能会遇到对您创建的任何实例的循环引用问题(其中 manager 实例保持事件状态,因为信号仍在引用该实例的绑定(bind)方法),或者您的信号处理程序过早清理的地方,因为您的绑定(bind)方法存储在弱引用中,因此不会有任何其他 引用以保持它们的活力。

查看 signalslot source code ,我看到项目的当前迭代使用了硬引用,所以你的 manager 实例永远不会被清除,除非你明确地这样做。仅出于这个原因,我会避免使用方法作为信号处理程序。看看using python WeakSet to enable a callback functionality如果您想改用弱引用。

关于python - 将装饰器类应用于类方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36403494/

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