weChatThread线程类

之前一直不会python多线程,写这个程序的时候,发现不用多线程会陷入无限未响应状态。于是学了半天python多线程,但是在主函数里写的时候,发现一个问题,Ui主线程和工作线程没有分离,使用itchat等库的时候会堵塞主线程,换句话说PyQt中子线程不能操作GUI界面。之前写的多线程仍然属于Ui主线程,是其子线程,所以才造成未响应。
既然知道问题了,那就查资料解决问题,后来,在几篇博客上找到了解决办法

然后仿照第一篇博客,重写了QThread类,并借鉴第三篇博客,学会了PyQt多线程中的信号/槽机制,用来传递参数。

#!/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()必须写在__init__前面,里面可接收的参数类型挺多的,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):
'''
重写run()函数,
:return:
'''
@itchat.msg_register([TEXT, PICTURE, MAP, CARD, SHARING, RECORDING, ATTACHMENT, VIDEO, FRIENDS], isFriendChat=True,
isGroupChat=True)
def getMsg(msg):
'''
注册消息类型,并对不同类型的消息执行不用的操作
: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'] # 消息发送人备注
elif itchat.search_friends(userName=msg['FromUserName'])['NickName']: # 消息发送人昵称
msg_from = itchat.search_friends(userName=msg['FromUserName'])['NickName'] # 消息发送人昵称
else:
msg_from = r"读取发送消息好友失败"
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__() + " 经度->" + 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):
'''
当消息类型为通知类的时候,查找消息内容是否为撤回消息,如果是,则执行撤回后的防撤回操作
: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 + ',撤回内容为:' + 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 #创建BuckUp文件夹
if not os.path.exists(".\\BackUp\\"):
os.mkdir('.\\BackUp\\')
#启动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+",程序运行时,请用手机扫描弹出的二维码进行登录,并确保电脑上自带的Window照片查"
"看器可用,撤回的图片文件等可下载附件连同运行日志保存在程序目录下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):
'''
往运行日志窗口写撤回消息的内容
:param msg:
:return:
'''
self.textBrowser.append(msg) def createTrayIcon(self):
'''
创建托盘图标,可以让程序最小化到windows托盘中运行
: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):
'''
为托盘图标添加功能
:return:
'''
self.restoreAction=QAction("恢复",self,triggered=self.showNormal)
self.quitAction=QAction("退出",self,triggered=QApplication.instance().quit) def iconActivated(self,reason):
'''
激活托盘功能
: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类实例化,并启动线程
:return:
'''
from weChatThread import weChatWord
self.wcBWThread=weChatWord()
self.wcBWThread.getMsgSignal.connect(self.setLog)
self.wcBWThread.start()

程序界面

程序界面仍然由Qtdesigner设计

后记

第一次尝试多线程编程,并且具体应用到实际项目中去,收获良多。

Python3.5+PyQt5多线程+itchat实现微信防撤回桌面版代码的更多相关文章

  1. 用Python教你微信防撤回(文本、图片、语音、视频、名片等...)

    大家在使用微信过程中,有时候消息还没看到,就被撤回了.毕竟好奇心大家都有,明知到消息被撤回了,就更想去看一下是什么内容心里想着万一是女神给我表白了呢.. 今天就用Python来做个微信防撤回的小功能. ...

  2. 我已经看到了,撤回也没用了(PC微信防撤回补丁)

    前两天看 GitHub 发现一个有趣的项目,PC微信防撤回补丁,本着研究学习的目的,在看过源码,一顿疯狂操作之后,了解了其原理是基于修改 wechatwin.dll 达到防撤回的. 于是乎,自己动手玩 ...

  3. 记录实践PC端微信防撤回实现过程(基于3.1.0.67版本)

    利用OD实现对PC端微信防撤回功能的实现 文章最后有一键补丁工具哦~ 准备工具 1.OD 2.PC微信客户端(3.1.0.67) 过程 1.运行微信客户端,不需要登录 2.运行OD,左上角选择附加进程 ...

  4. PC端微信防撤回功能分析

    1.打开PC端微信的安装目录,有一个WeChatWin.dll文件,微信的所有功能基本上都在这个文件中了 2.OD打开,搜索字符串revokemsg(撤回消息,掌握一门外语是多么的重要啊!!!),在所 ...

  5. [软件逆向]实战Mac系统下的软件分析+Mac QQ和微信的防撤回

      0x00  一点废话 最近因为Mac软件收费的比较多,所以买了几款正版软件,但是有的软件卖的有点贵,买了感觉不值,不买吧,又觉得不方便,用别人的吧,又怕不安全.于是我就买了正版的Hopper Di ...

  6. Python3 itchat实现微信定时发送群消息

    Python3 itchat实现微信定时发送群消息 一.简介 1,使用微信,定时往指定的微信群里发送指定信息. 2,需要发送的内容使用excel进行维护,指定要发送的微信群名.时间.内容. 二.py库 ...

  7. 基于itchat的微信群聊小助手基础开发(一)

    前段时间由于要管理微信群,基于itchat开发了一个简单的微信机器人 主要功能有: 图灵机器人功能 群聊昵称格式修改提示 消息防撤回功能 斗图功能 要开发一个基于itchat的最基本的聊天机器人,在g ...

  8. itchat库微信自动回复祝福语

    过年了,之前看到一些python文章介绍用itchat自动回复微信,我自己就写了一个. 官方文档https://itchat.readthedocs.io/zh/latest/,这个库挺简洁的,对着接 ...

  9. Python_20行代码实现微信消息防撤回(简易版)

    学习了一下如何用python实现微信消息的防撤回, 主要思路就是: 时时监控微信,将对方发送的消息缓存下来 如果对方撤回了消息,就将该缓存信息发送给文件传输助手 但其实这功能,基本上毫无意义,看到别人 ...

随机推荐

  1. 开发中清除css加载的缓存使用

  2. 如何写一个优秀的GitHub项目README文档?

    今天给大家介绍一个Github上的README文档写作教程模版,该模版目前获得6634颗星星,2296Fork,相对而言,还是比较得到大家认可的.不花哨,不别出心裁,一个比较实用的,普适性的架子:所谓 ...

  3. Ubuntu下安装eclipse遇到的问题

    今天在Ubuntu中安装eclipse时遇到如下问题: 解决方法: 打开eclipse安装目录下eclipse.ini文件 在文件最开头(注:一定是最开头)加上如下语句(-startup前面两行),第 ...

  4. Xgboost调参总结

    一.参数速查 参数分为三类: 通用参数:宏观函数控制. Booster参数:控制每一步的booster(tree/regression). 学习目标参数:控制训练目标的表现. 二.回归 from xg ...

  5. WebAPI的跨域访问CORS三种方法

    跨域访问: JSONP的原理利用<script>没有跨域访问的限制,利用<script>的src跨域访问api,api会根据数据把json包装在一个js里面,这样跨域的客户端拿 ...

  6. iOS 第三方框架-Masonry

    介绍地址:http://www.cocoachina.com/ios/20141219/10702.html 官网:https://github.com/SnapKit/Masonry 记住:一定要先 ...

  7. VS.C#如何向数据数据库中存入和读取图片的

    写入图片部分代码:假设图片为 test.gifbyte [] bytes = File.ReadAllBytes(@"c:\test.gif");SqlConnection con ...

  8. laravel 前后端分离 token

    由于自己开发的项目中用到了 JWT 技术,前端采用了 Vue.js 框架,后端采用了 CodeIgniter 框架,故作此文帮助使用相同技术栈的朋友们. 具体思路如下:把后端生成的 JWT token ...

  9. html中通过js获取接口JSON格式数据解析以及跨域问题

    前言:本人自学前端开发,一直想研究下js获取接口数据在html的实现,顺利地找到了获取数据的方法,但是有部分接口在调用中出现无法展示数据.经查,发现时跨域的问题,花费了一通时间,随笔记录下过程,以方便 ...

  10. day1 hbuilder的使用

    一.互联网的原理 1.概述 html:用来制作网页. 互联网原理:上网即请求数据. 用户通过在浏览器上输入一个网址,通过HTTP协议向服务器发送请求,服务器做出响应,将相关的网页数据传输到本地计算机, ...