私はネットワークスレッドを持っているスレッド化されたアプリケーションを持っています。UI 部分は、callback
このスレッドに a を渡します。スレッドは通常のpython スレッドです - NO QThread
です。
このスレッド内で PyQT スロットを発行することは可能ですか?
私はネットワークスレッドを持っているスレッド化されたアプリケーションを持っています。UI 部分は、callback
このスレッドに a を渡します。スレッドは通常のpython スレッドです - NO QThread
です。
このスレッド内で PyQT スロットを発行することは可能ですか?
いいえ、このような Python スレッドから PyQt シグナルを送信することはできません。
ただし、可能な解決策は、両方のスレッドで共有される追加のオブジェクトを使用して、最終的にスレッドセーフな PyQt シグナルを発行するために必要な操作を行うことです。
これは「SafeConnector」クラスの実装で、接続されたソケットのペアと Queue を使用して 2 つのスレッド間でデータを交換し、QSocketNotifier を使用して Qt のループに戻ります。QObject は、適切な Qt シグナルを発行できるようにするために使用されます。
from PyQt4 import Qt, QtCore, QtGui
import threading
import socket
import Queue
import time
# Object of this class has to be shared between
# the two threads (Python and Qt one).
# Qt thread calls 'connect',
# Python thread calls 'emit'.
# The slot corresponding to the emitted signal
# will be called in Qt's thread.
class SafeConnector:
def __init__(self):
self._rsock, self._wsock = socket.socketpair()
self._queue = Queue.Queue()
self._qt_object = QtCore.QObject()
self._notifier = QtCore.QSocketNotifier(self._rsock.fileno(),
QtCore.QSocketNotifier.Read)
self._notifier.activated.connect(self._recv)
def connect(self, signal, receiver):
QtCore.QObject.connect(self._qt_object, signal, receiver)
# should be called by Python thread
def emit(self, signal, args):
self._queue.put((signal, args))
self._wsock.send('!')
# happens in Qt's main thread
def _recv(self):
self._rsock.recv(1)
signal, args = self._queue.get()
self._qt_object.emit(signal, args)
class PythonThread(threading.Thread):
def __init__(self, connector, *args, **kwargs):
threading.Thread.__init__(self, *args, **kwargs)
self.connector = connector
self.daemon = True
def emit_signal(self):
self.connector.emit(QtCore.SIGNAL("test"), str(time.time()))
def run(self):
while True:
time.sleep(1)
self.emit_signal()
if __name__ == '__main__':
app = QtGui.QApplication([])
mainwin = QtGui.QMainWindow()
label = QtGui.QLabel(mainwin)
mainwin.setCentralWidget(label)
connector = SafeConnector()
python_thread = PythonThread(connector)
connector.connect(QtCore.SIGNAL("test"), label.setText)
python_thread.start()
mainwin.show()
app.exec_()