詳解pyqt5 動畫在QThread線程中無法運行問題
自己做了一個tcp工具,在學(xué)習(xí)動畫的時候踩了坑,需求是根據(jù)上線變綠色,離線變灰色,如果連接斷開了,則變?yōu)榛疑?/p>
問題現(xiàn)象:

可以看到點擊“連接”,“離線”的時候動畫是正常的,但是當(dāng)tcp超時斷開后,雖然離線按鈕變?yōu)檫B接了,卻沒有執(zhí)行離線動畫
關(guān)鍵源代碼如下
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: # 十六進(jìn)制數(shù)據(jù)處理,兩個字節(jié)隔開
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()
問題點:
經(jīng)過谷爹搜索,終于找到了問題原因詳見https://stackoverflow.com/questions/44328750/pyqt-qgraphicscene-move-item-in-background-thread
大致原因就是QGraphics Scene 不是一個安全的線程對象,我們不能直接在線程中去改變主程序的狀態(tài),我們必須通過信號的方式去更新QGraphics
解決方法:
首先,我們編輯一個信號方法
def threadAnimate(self, message):
if message == "1":
self.scene.offlineCol.start()
然后添加相關(guān)信號槽
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()
在線程中發(fā)出離線動畫的信號
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: # 十六進(jìn)制數(shù)據(jù)處理,兩個字節(jié)隔開
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多線程收發(fā)消息原理一樣

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
python的tkinter、socket庫開發(fā)tcp的客戶端和服務(wù)端詳解
本文介紹了TCP通訊流程和開發(fā)步驟,包括客戶端和服務(wù)端的實現(xiàn),客戶端使用Python的tkinter庫實現(xiàn)圖形化界面,服務(wù)端使用socket庫監(jiān)聽連接并處理消息,文章還提供了客戶端和服務(wù)端的代碼示例2025-01-01
python dataclass 快速創(chuàng)建數(shù)據(jù)類的方法
在Python中,dataclass是一種用于快速創(chuàng)建數(shù)據(jù)類的裝飾器和工具,本文實例代碼中我們定義了一個Person數(shù)據(jù)類,并使用fields()函數(shù)遍歷其字段,打印出每個字段的名稱、類型、默認(rèn)值和元數(shù)據(jù),對python dataclass 數(shù)據(jù)類相關(guān)知識感興趣的朋友一起看看吧2024-03-03
Python?中如何使用requests模塊發(fā)布表單數(shù)據(jù)
requests 庫是 Python 的主要方面之一,用于創(chuàng)建對已定義 URL 的 HTTP 請求,本篇文章介紹了 Python requests 模塊,并說明了我們?nèi)绾问褂迷撃K在 Python 中發(fā)布表單數(shù)據(jù),感興趣的朋友跟隨小編一起看看吧2023-06-06
Django filter動態(tài)過濾與排序?qū)崿F(xiàn)過程解析
這篇文章主要介紹了Django filter動態(tài)過濾與排序?qū)崿F(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-11-11
python dict remove數(shù)組刪除(del,pop)
我們在用數(shù)組列表做刪除的時候,可能選擇2個方法,一個是del,一個是pop方法2013-03-03

