时间:2021-05-22
自己做了一个tcp工具,在学习动画的时候踩了坑,需求是根据上线变绿色,离线变灰色,如果连接断开了,则变为灰色
问题现象:
可以看到点击“连接”,“离线”的时候动画是正常的,但是当tcp超时断开后,虽然离线按钮变为连接了,却没有执行离线动画
关键源代码如下
class BSJTcpThread(QtCore.QThread): recv_signal = QtCore.pyqtSignal(str) send_signal = QtCore.pyqtSignal(str) def __init__(self, socketcp, onBtn, heartcheck, senBtn, scene): super().__init__() self.s = socketcp self.yqtool = Bianlifunction() self.onBtn = onBtn self.heartcheck = heartcheck self.sendBtn = senBtn self.scene1 = scene def run(self): """线程""" global stopsingle stopsingle = 0 while 1: btcpreceive = self.s.recv(1024) tcpreceive1 = str(binascii.b2a_hex(btcpreceive), encoding="utf-8") tcpreceive = "" i = 0 while i < len(tcpreceive1) - 1: # 十六进制数据处理,两个字节隔开 if i == len(tcpreceive1) - 2: tcpreceive += tcpreceive1[i:i + 2] i += 2 else: tcpreceive += tcpreceive1[i:i + 2] + " " i += 2 if tcpreceive == "": stopsingle = 1 self.s.shutdown(2) self.s.close() self.onBtn.setText("连接") self.scene1.offlineCol.start() # 启动离线动画 self.heartcheck.setChecked(False) self.heartcheck.setVisible(False) self.sendBtn.setDisabled(True) else: self.recv_signal.emit(tcpreceive) if stopsingle == 1: break然后再启动线程
self.tcpth = BSJTcpThread(self.s, self.onBtn, self.heartcheck, self.sendBtn, self.scene) self.tcpth.recv_signal.connect(self.fillrecvmsg) self.tcpth.send_signal.connect(self.fillsendmsg) self.tcpth.start()问题点:
经过谷爹搜索,终于找到了问题原因详见https://stackoverflow.com/questions/44328750/pyqt-qgraphicscene-move-item-in-background-thread
大致原因就是QGraphics Scene 不是一个安全的线程对象,我们不能直接在线程中去改变主程序的状态,我们必须通过信号的方式去更新QGraphics
解决方法:
首先,我们编辑一个信号方法
def threadAnimate(self, message): if message == "1": self.scene.offlineCol.start()然后添加相关信号槽
self.tcpth = BSJTcpThread(self.s, self.onBtn, self.heartcheck, self.sendBtn) self.tcpth.recv_signal.connect(self.fillrecvmsg) self.tcpth.send_signal.connect(self.fillsendmsg) self.tcpth.animate_signal.connect(self.threadAnimate) # 添加一个动画信号 self.tcpth.start()在线程中发出离线动画的信号
class BSJTcpThread(QtCore.QThread): recv_signal = QtCore.pyqtSignal(str) send_signal = QtCore.pyqtSignal(str) animate_signal = QtCore.pyqtSignal(str) def __init__(self, socketcp, onBtn, heartcheck, senBtn): super().__init__() self.s = socketcp self.yqtool = Bianlifunction() self.onBtn = onBtn self.heartcheck = heartcheck self.sendBtn = senBtn def run(self): """线程""" global stopsingle stopsingle = 0 while 1: btcpreceive = self.s.recv(1024) tcpreceive1 = str(binascii.b2a_hex(btcpreceive), encoding="utf-8") tcpreceive = "" i = 0 while i < len(tcpreceive1) - 1: # 十六进制数据处理,两个字节隔开 if i == len(tcpreceive1) - 2: tcpreceive += tcpreceive1[i:i + 2] i += 2 else: tcpreceive += tcpreceive1[i:i + 2] + " " i += 2 if tcpreceive == "": stopsingle = 1 self.s.shutdown(2) self.s.close() self.onBtn.setText("连接") self.animate_signal.emit("1") self.heartcheck.setChecked(False) self.heartcheck.setVisible(False) self.sendBtn.setDisabled(True) else: self.recv_signal.emit(tcpreceive) if stopsingle == 1: break然后就可以了,这个和QThread多线程收发消息原理一样
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
PyQt5简单多线程信号与槽的使用最简单的多线程使用方法是利用QThread函数,展示QThread函数和信号简单结合的方法importsysfromPyQt5
在PyQt5中使用QThread的时候,要注意把所有QThread的对象在主类中的init(或者放在所有类函数的外面)中进行实例化,不然可能在多个QThread
PyCharm不识别PyQt5的问题如图所示,引用PyQt5的时候显示错误“ModuleNotFoundError:Nomodulenamed'pyqt5'”首
用PyQt中的QThread类实现多线程利用PyQt中的pyqtSignal类实现信息的触发和捕获,即定义事件和订阅事件利用QThread类实现多线程的方法1,
PyQt中的线程类QtCore.QThread,使用时继承QThread类启动界面的线程暂称为UI线程。界面执行命令时都在自己的UI线程中。如果在UI线程中执行