2

我正在尝试编写通用单发插槽包装器。这是我现在拥有的:

def single_shot_connect(signal, slot):
  signaled = [False]
  def single_shot_slot(*args, **kwargs):
    if signaled[0]: return
    signaled[0] = True
    signal.disconnect(single_shot_slot)
    slot(*args, **kwargs)

  signal.connect(single_shot_slot)

def foo(): pass

class Bar(QtCore.QObject): sig = QtCore.Signal([int])

bar1 = Bar()
bar1.sig.connect(foo)
bar1.sig.emit(1)  # (1)

bar2 = Bar()
single_shot_connect(bar2.sig, foo)
bar2.sig.emit(1)  # (2)

问题是标记(2)的代码不起作用,因为它在正常工作时尝试foo()使用一个参数进行调用(1)

我怎样才能single_shot_connect()工作?

编辑: single_shot_connect()可以针对这种特殊情况进行修复,如下所示:

def single_shot_connect(signal, slot):
  signaled = [False]
  def single_shot_slot(): # Note absence of *args, **kwargs
    if signaled[0]: return
    signaled[0] = True
    signal.disconnect(single_shot_slot)
    slot() # Note absence of *args, **kwargs

  signal.connect(single_shot_slot)

但这对我来说并不有趣,因为我想要single_shot_connect()通用且可用于任何类型的信号和插槽。

4

1 回答 1

3

由于 Qt保证调用的槽的顺序将是connect语句的顺序,您可以简单地这样做:

def single_shot_connect(signal, slot):
    def disconnect(*args, **kwargs):
        signal.disconnect(slot)
        signal.disconnect(disconnect)
    signal.connect(slot)
    signal.connect(disconnect)

signal发出后,slot将被调用。它后面跟着一个disconnect调用,它删除了slotand 本身。因此,它们不会被调用以进行后续发射。

完整使用示例:

import sys
from PySide import QtCore

class Foo(QtCore.QObject):
    sig = QtCore.Signal(int)

def bar():
    print 'running bar'


def single_shot_connect(signal, slot):
    def disconnect(*args, **kwargs):
        signal.disconnect(slot)
        signal.disconnect(disconnect)
    signal.connect(slot)
    signal.connect(disconnect)



if __name__ == '__main__':
    app = QtCore.QCoreApplication(sys.argv)

    foo = Foo()
    single_shot_connect(foo.sig, bar)

    foo.sig.emit(1)
    foo.sig.emit(2)

    QtCore.QTimer.singleShot(1000, app.exit)

    sys.exit(app.exec_())
于 2013-01-10T05:33:48.420 回答