Python 在信号/插槽中使用自定义对象(类似PyQt_PyObject)

Python 在信号/插槽中使用自定义对象(类似PyQt_PyObject),python,pyside,Python,Pyside,由于许可,我设法使用PySide而不是PyQt 我需要使用信号/插槽机制在线程之间传递自定义对象。对于PyQt,我可以使用PyQt\u PyObject类型作为信号参数,但显然,PySide中不存在这种类型: TypeError: Unknown type used to call meta function (that may be a signal): PyQt_PyObject 我尝试使用object而不是PyQt\u PyObject,但只有在信号和插槽之间使用DirectConnec

由于许可,我设法使用PySide而不是PyQt

我需要使用信号/插槽机制在线程之间传递自定义对象。对于PyQt,我可以使用
PyQt\u PyObject
类型作为信号参数,但显然,PySide中不存在这种类型:

TypeError: Unknown type used to call meta function (that may be a signal): PyQt_PyObject
我尝试使用
object
而不是
PyQt\u PyObject
,但只有在信号和插槽之间使用DirectConnection类型时才会发生这种情况:

self.connect(dummyEmitter,
             QtCore.SIGNAL("logMsgPlain(object)"),
             self._logMsgPlain,
             QtCore.Qt.DirectConnection)
使用QueuedConnection,我得到一个错误:

QObject::connect: Cannot queue arguments of type 'object'
(Make sure 'object' is registered using qRegisterMetaType().)
我说“事情会发生”,因为到目前为止它还不起作用。由于DirectConnection类型,我现在收到错误:

QObject::startTimer: timers cannot be started from another thread
QPixmap: It is not safe to use pixmaps outside the GUI thread
etc ...
我该怎么办? PySide中是否有类似的PyQt_PyObject类型

编辑: 此小示例将失败:

from PySide import QtCore, QtGui
import sys

class Object(QtCore.QObject):
    ''' A dummy emitter that send a list to the thread '''
    def emitSignal(self):
        someList = [0, 1, 2, 3]
        self.emit(QtCore.SIGNAL("aSignal(object)"), someList)

class Worker(QtCore.QObject):
    def aSlot(self, value):
        print "List: {}".format(value)

if __name__ == "__main__":
    app = QtGui.QApplication(sys.argv)

    worker = Worker()
    obj = Object()

    thread = QtCore.QThread()
    worker.moveToThread(thread)
    QtCore.QObject.connect(obj, QtCore.SIGNAL("aSignal(object)"), worker.aSlot)
    # The exemple will pass with the line below uncommented
    # But obviously, I can't use a DirectConnection with a worker thread and the GUI thread
    # QtCore.QObject.connect(obj, QtCore.SIGNAL("aSignal(object)"), worker.aSlot, QtCore.Qt.DirectConnection)

    thread.start()
    obj.emitSignal()

    app.exec_()

目前,我找到的唯一解决方案是切换到新样式的信号/插槽语法:

from PySide import QtCore, QtGui
import sys

class Object(QtCore.QObject):
    aSignal = QtCore.Signal(object)
    def emitSignal(self):
        someList = [0, 1, 2, 3]
        self.aSignal.emit(someList)

class Worker(QtCore.QObject):
    def aSlot(self, value):
        print "List: {}".format(value)

if __name__ == "__main__":
    app = QtGui.QApplication(sys.argv)

    worker = Worker()
    obj = Object()

    thread = QtCore.QThread()
    worker.moveToThread(thread)
    obj.aSignal.connect(worker.aSlot)

    thread.start()
    obj.emitSignal()

    app.exec_()

但我很想知道是否有使用旧式语法的解决方案,但目前似乎没有。

目前,我找到的唯一解决方案是切换到新样式的信号/插槽语法:

from PySide import QtCore, QtGui
import sys

class Object(QtCore.QObject):
    aSignal = QtCore.Signal(object)
    def emitSignal(self):
        someList = [0, 1, 2, 3]
        self.aSignal.emit(someList)

class Worker(QtCore.QObject):
    def aSlot(self, value):
        print "List: {}".format(value)

if __name__ == "__main__":
    app = QtGui.QApplication(sys.argv)

    worker = Worker()
    obj = Object()

    thread = QtCore.QThread()
    worker.moveToThread(thread)
    obj.aSignal.connect(worker.aSlot)

    thread.start()
    obj.emitSignal()

    app.exec_()

但我很想知道是否有一种使用旧式语法的解决方案,但目前看来,似乎没有。

您读过本页吗。我总是使用“新式”信号/插槽语法,并且能够在线程之间传递Python对象,前提是它们可以被pickle。是的,我读了这一页,但我没有看到PyQt和PySide在这一级别上的任何差异。你能发布一个失败的小示例吗?PySide当然可以做你想做的,但这里没有足够的信息让我进一步评论。我从未在任何PySide程序中声明过连接类型-如果两个对象位于不同的QThread中,它知道该做什么。我也从不声明连接类型,但我注意到它与直接连接一起工作(显然,在我的例子中,我不能使用这种类型的连接,它应该排队,但PySide告诉我类型为
object
的参数不能排队)。该示例通过将
object
类型替换为
PyQt\u PyObject
来使用PyQt,您是否阅读了本页。我总是使用“新式”信号/插槽语法,并且能够在线程之间传递Python对象,前提是它们可以被pickle。是的,我读了这一页,但我没有看到PyQt和PySide在这一级别上的任何差异。你能发布一个失败的小示例吗?PySide当然可以做你想做的,但这里没有足够的信息让我进一步评论。我从未在任何PySide程序中声明过连接类型-如果两个对象位于不同的QThread中,它知道该做什么。我也从不声明连接类型,但我注意到它与直接连接一起工作(显然,在我的例子中,我不能使用这种类型的连接,它应该排队,但PySide告诉我类型为
object
的参数不能排队)。该示例通过将
object
类型替换为
PyQt\u PyObject
来使用PyQt。解决方案是永远不要使用旧式语法,它难看、不发音且容易出错。解决方案是永远不要使用旧式语法,它难看、不发音且容易出错。