weChatThread線程類
之前一直不會(huì)python多線程,寫(xiě)這個(gè)程序的時(shí)候,發(fā)現(xiàn)不用多線程會(huì)陷入無(wú)限未響應(yīng)狀態(tài)。于是學(xué)了半天python多線程,但是在主函數(shù)里寫(xiě)的時(shí)候,發(fā)現(xiàn)一個(gè)問(wèn)題,Ui主線程和工作線程沒(méi)有分離,使用itchat等庫(kù)的時(shí)候會(huì)堵塞主線程,換句話說(shuō)PyQt中子線程不能操作GUI界面。之前寫(xiě)的多線程仍然屬于Ui主線程,是其子線程,所以才造成未響應(yīng)。
既然知道問(wèn)題了,那就查資料解決問(wèn)題,后來(lái),在幾篇博客上找到了解決辦法
然后仿照第一篇博客,重寫(xiě)了QThread類,并借鑒第三篇博客,學(xué)會(huì)了PyQt多線程中的信號(hào)/槽機(jī)制,用來(lái)傳遞參數(shù)。
下面貼代碼
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
__author__ = 'memgq'
from PyQt5.QtCore import QThread,pyqtSignal
import itchat
import time,os
import shutil
import re
from itchat.content import *
class weChatWord(QThread):
getMsgSignal = pyqtSignal(str) #pyqtSignal()必須寫(xiě)在__init__前面,里面可接收的參數(shù)類型挺多的,str,list,dict都支持
def __init__(self,parent=None):
super(weChatWord,self).__init__(parent)
self.msg_list=[]
self.type_list=['Picture','Recording', 'Attachment','Video']
def clearList(self):
'''
清空緩存消息和文件
:return:
'''
tm_now=time.time()
len_list=len(self.msg_list)
if len_list>0:
for i in range(len_list):
if tm_now-self.msg_list[i]['msg_time']>121:
if self.msg_list[i]['msg_type'] in self.type_list:
try:
os.remove(".\\BackUp\\"+self.msg_list[i]['msg_content'])
except Exception as e:
print(e)
finally:
pass
else:break
self.msg_list=self.msg_list[i:]
def run(self):
'''
重寫(xiě)run()函數(shù),
:return:
'''
@itchat.msg_register([TEXT, PICTURE, MAP, CARD, SHARING, RECORDING, ATTACHMENT, VIDEO, FRIENDS], isFriendChat=True,
isGroupChat=True)
def getMsg(msg):
'''
注冊(cè)消息類型,并對(duì)不同類型的消息執(zhí)行不用的操作
:param msg:
:return:
'''
msg_dict={}
# pprint.pprint(msg)
msg_id = msg['MsgId'] # 消息ID
msg_time = msg['CreateTime']
msg_url=None
msg_group=""
if (itchat.search_friends(userName=msg['FromUserName'])):
if itchat.search_friends(userName=msg['FromUserName'])['RemarkName']:
msg_from = itchat.search_friends(userName=msg['FromUserName'])['RemarkName'] # 消息發(fā)送人備注
elif itchat.search_friends(userName=msg['FromUserName'])['NickName']: # 消息發(fā)送人昵稱
msg_from = itchat.search_friends(userName=msg['FromUserName'])['NickName'] # 消息發(fā)送人昵稱
else:
msg_from = r"讀取發(fā)送消息好友失敗"
else:
msg_group = msg['User']['NickName']
msg_from = msg['ActualNickName']
msg_type = msg['Type']
if msg_type in ['Text', 'Friends','Sharing']:
msg_content = msg['Text']
msg_url = msg['Url']
elif msg_type in self.type_list:
msg_content=msg['FileName']
msg['Text'](msg['FileName'])
shutil.move(msg_content,r'.\\BackUp\\')
elif msg['Type'] == 'Card':
msg_content = msg['RecommendInfo']['NickName'] + r" 的名片"
elif msg['Type'] == 'Map':
x, y, location = re.search("<location x=\"(.*?)\" y=\"(.*?)\".*label=\"(.*?)\".*",
msg['OriContent']).group(1,
2,
3)
if location is None:
msg_content = r"緯度->" + x.__str__() + " 經(jīng)度->" + y.__str__()
else:
msg_content = r"" + location
msg_dict={'msg_id':msg_id,'msg_time':msg_time,'msg_from':msg_from,'msg_group':msg_group,
'msg_content':msg_content,'msg_type':msg_type,'msg_url':msg_url}
self.msg_list.append(msg_dict)
self.clearList()
@itchat.msg_register([NOTE],isFriendChat=True, isGroupChat=True)
def recall(msg):
'''
當(dāng)消息類型為通知類的時(shí)候,查找消息內(nèi)容是否為撤回消息,如果是,則執(zhí)行撤回后的防撤回操作
:param msg:
:return:
'''
# pprint.pprint(msg)
msg_content=msg['Content']
if re.search(r'\<replacemsg\>\<!\[CDATA\[(.*)撤回了一條消息\]\]\>\<\/replacemsg\>',msg_content):
msg_note=re.search(r'\<replacemsg\>\<!\[CDATA\[(.*)\]\]\>\<\/replacemsg\>',msg_content).group(1)
old_msg_id=re.search(r'\<msgid\>([0-9]+)\</msgid\>',msg_content).group(1)
for each in self.msg_list:
if each['msg_id']==old_msg_id:
timeArray = time.localtime()
otherStyleTime = time.strftime("%Y-%m-%d %H:%M:%S,", timeArray)
msg_note = msg_note + ',撤回內(nèi)容為:' + each['msg_content']
if each['msg_group']!='':
msg_note = "群組("+each['msg_group']+")中"+msg_note
msg_note=otherStyleTime+msg_note
itchat.send(msg_note,toUserName='filehelper')
self.msg_list.pop(self.msg_list.index(each))
self.getMsgSignal.emit(msg_note)
break
#創(chuàng)建BuckUp文件夾
if not os.path.exists(".\\BackUp\\"):
os.mkdir('.\\BackUp\\')
#啟動(dòng)itchat()
itchat.auto_login(hotReload=True)
itchat.run()
主程序類
class mainwindowapp(QMainWindow,wechatunrecall.Ui_MainWindow):
def __init__(self):
super().__init__()
self.setupUi(self)
self.createActions()
self.createTrayIcon()
self.pushButton.clicked.connect(self.saveLog)
self.pushButton_2.clicked.connect(self.clearlog)
self.pushButton_3.clicked.connect(self.houtai)
self.trayIcon.activated.connect(self.iconActivated)
timeArray = time.localtime()
otherStyleTime = time.strftime("%Y-%m-%d %H:%M:%S", timeArray)
self.setLog(otherStyleTime+",程序運(yùn)行時(shí),請(qǐng)用手機(jī)掃描彈出的二維碼進(jìn)行登錄,并確保電腦上自帶的Window照片查"
"看器可用,撤回的圖片文件等可下載附件連同運(yùn)行日志保存在程序目錄下BackUp文件夾中。\n")
self.weChatBigWord()
def saveLog(self):
'''
保存日志
:return:
'''
if not os.path.exists(".\\BackUp\\"):
os.mkdir(".\\BackUp\\")
timeArray = time.localtime()
otherStyleTime = time.strftime("%Y-%m-%d%H%M%S", timeArray)
text=self.textBrowser.toPlainText()
logPath=".\\BackUp\\"+otherStyleTime+'.txt'
with open(logPath,'w') as f:
f.write(text)
def setLog(self,msg):
'''
往運(yùn)行日志窗口寫(xiě)撤回消息的內(nèi)容
:param msg:
:return:
'''
self.textBrowser.append(msg)
def createTrayIcon(self):
'''
創(chuàng)建托盤(pán)圖標(biāo),可以讓程序最小化到windows托盤(pán)中運(yùn)行
:return:
'''
self.trayIconMenu=QMenu(self)
self.trayIconMenu.addAction(self.restoreAction)
self.trayIconMenu.addSeparator()
self.trayIconMenu.addAction(self.quitAction)
self.trayIcon=QSystemTrayIcon(self)
self.trayIcon.setContextMenu(self.trayIconMenu)
self.trayIcon.setIcon(QIcon('./media/images/maincion.png'))
self.setWindowIcon(QIcon('./media/images/maincion.png'))
self.trayIcon.show()
def createActions(self):
'''
為托盤(pán)圖標(biāo)添加功能
:return:
'''
self.restoreAction=QAction("恢復(fù)",self,triggered=self.showNormal)
self.quitAction=QAction("退出",self,triggered=QApplication.instance().quit)
def iconActivated(self,reason):
'''
激活托盤(pán)功能
:param reason:
:return:
'''
if reason in (QSystemTrayIcon.Trigger, QSystemTrayIcon.DoubleClick):
self.showNormal()
def houtai(self):
self.hide()
def clearlog(self):
self.textBrowser.clear()
def weChatBigWord(self):
'''
weChatThread類實(shí)例化,并啟動(dòng)線程
:return:
'''
from weChatThread import weChatWord
self.wcBWThread=weChatWord()
self.wcBWThread.getMsgSignal.connect(self.setLog)
self.wcBWThread.start()
程序界面
程序界面仍然由Qtdesigner設(shè)計(jì)
后記
第一次嘗試多線程編程,并且具體應(yīng)用到實(shí)際項(xiàng)目中去,收獲良多。
最近打算學(xué)點(diǎn)Django,有沒(méi)有推薦教程?