Python3.5+PyQt5多線程+itchat實(shí)現(xiàn)微信防撤回桌面版代碼(二)

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)有推薦教程?

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,578評(píng)論 6 544
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,701評(píng)論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 178,691評(píng)論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 63,974評(píng)論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,694評(píng)論 6 413
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 56,026評(píng)論 1 329
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,015評(píng)論 3 450
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 43,193評(píng)論 0 290
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,719評(píng)論 1 336
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,442評(píng)論 3 360
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,668評(píng)論 1 374
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,151評(píng)論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,846評(píng)論 3 351
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 35,255評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 36,592評(píng)論 1 295
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,394評(píng)論 3 400
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,635評(píng)論 2 380

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,771評(píng)論 25 708
  • 【養(yǎng)心養(yǎng)意學(xué)習(xí)力踐行Day2】補(bǔ) ^o^晨起英語(yǔ)喚醒 ^o^在助教陪同下讀了古詩(shī) ^o^畫(huà)畫(huà) ^o^睡前故事每天都...
    愛(ài)己及人閱讀 124評(píng)論 0 0
  • 第一次在網(wǎng)上寫(xiě)日記哦,當(dāng)然這里不算在qq空間的日志。一直很糾結(jié)。想要寫(xiě)日記,但是估計(jì)是握筆姿勢(shì)不對(duì)吧還是自己身體孱...
    楊桃閱讀 461評(píng)論 4 1
  • 今天下午為了年底業(yè)績(jī)沖刺,我所在事業(yè)部老大帶著我們團(tuán)隊(duì)找了一家迪歐咖啡,專門在寬松的環(huán)境下討論了近四個(gè)小時(shí),目的是...
    海闊憑躍躍閱讀 319評(píng)論 0 4
  • 一江寒水照殘?jiān)疲麊?wèn)晚風(fēng)尋故人。 催舟撒網(wǎng)斜陽(yáng)伴,一縷枯草幾縷愁。
    漁舢閱讀 508評(píng)論 0 1