当前位置: 首页 > news >正文

pyqt5 动画在QThread线程中无法运行问题

自己做了一个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多线程收发消息原理一样

 

转载于:https://www.cnblogs.com/semishigure/p/8986088.html

相关文章:

  • Spring Boot 定时任务
  • mybatis-generator逆向工程生成boolean字段解决办法
  • MySQL 高可用性keepalived+mysql
  • 今天来尝个鲜——vCenter6.7 + ESXi6.7
  • springcloud微服务学习之ribbon
  • 如何将SAP Multi Target应用部署到SAP云平台的Cloud Foundry环境去
  • solr cloud搭建
  • 5月7日云栖精选夜读丨如何用阿里云快速构建游戏发行技术体系
  • Java基础-MySQL数据库扫盲篇
  • Spring Boot项目中发送邮件
  • 使用Leangoo做销售管理系统
  • 用ASP.NET Core 2.0 建立规范的 REST API -- 预备知识
  • Hive数据导入导出的几种方式
  • 字符串,json对象和json字符串
  • react 代码优化(一) ——事件处理
  • 【跃迁之路】【669天】程序员高效学习方法论探索系列(实验阶段426-2018.12.13)...
  • 2017 年终总结 —— 在路上
  • docker容器内的网络抓包
  • Fabric架构演变之路
  • JavaScript学习总结——原型
  • node入门
  • Spark VS Hadoop:两大大数据分析系统深度解读
  • SQLServer之创建数据库快照
  • 基于阿里云移动推送的移动应用推送模式最佳实践
  • 理解在java “”i=i++;”所发生的事情
  • 深入体验bash on windows,在windows上搭建原生的linux开发环境,酷!
  • 吐槽Javascript系列二:数组中的splice和slice方法
  • 走向全栈之MongoDB的使用
  • MiKTeX could not find the script engine ‘perl.exe‘ which is required to execute ‘latexmk‘.
  • ​总结MySQL 的一些知识点:MySQL 选择数据库​
  • $.ajax()方法详解
  • (3)(3.2) MAVLink2数据包签名(安全)
  • (C语言)字符分类函数
  • (附源码)springboot美食分享系统 毕业设计 612231
  • (附源码)SSM环卫人员管理平台 计算机毕设36412
  • (附源码)计算机毕业设计ssm基于B_S的汽车售后服务管理系统
  • (十六)串口UART
  • (转)大道至简,职场上做人做事做管理
  • .apk 成为历史!
  • .NET Standard 支持的 .NET Framework 和 .NET Core
  • .net 重复调用webservice_Java RMI 远程调用详解,优劣势说明
  • .NET/C# 在代码中测量代码执行耗时的建议(比较系统性能计数器和系统时间)...
  • @SuppressWarnings(unchecked)代码的作用
  • []AT 指令 收发短信和GPRS上网 SIM508/548
  • [3300万人的聊天室] 作为产品的上游公司该如何?
  • [BZOJ3223]文艺平衡树
  • [Django 0-1] Core.Checks 模块
  • [English]英语积累本
  • [IE技巧] 让IE 以全屏模式启动
  • [NAND Flash 7.1] 闪存系统性能优化方向集锦?AC timing? Cache? 多路并发?
  • [node] Node.js 缓冲区Buffer
  • [Oh My C++ Diary]用cout输出流保留输出小数位数的方法
  • [Raspberry Pi] Raspberry Pi 4配置OpenCV4.6.0和ncnn环境(32-bit operation system)
  • [RK3568 Android11] Input UI 使用流程
  • [RQNOJ 696] 【树形DP】