不知道昨天大家是怎

这篇不是总结文,题外话不多说。上一期我们知道怎么记录好友发来的信息,这里我们来看看怎么知道哪些消息是被撤回的,并且把它们转发给“文件传输助手”。

优化缓存机制

前面我们把好友发来的所有信息都存入到一个叫log的字典对象中,以免错过好友的任何一条信息。但是这么多信息都存储在里面,没有办法一条条的去分析到底是不是被撤回的信息,必须想办法把有用的信息挑选出来。

值得一提的是,微信撤回机制有一个规则,用户只能撤回2分钟内自己发出去的消息。所以相对每次接收到信息的时刻,log中2分钟以前的信息是不能被撤回的。这些信息属于无用信息,完全可以删除,也避免了信息内容过多无法分析和程序运行效率因信息占用过多内存而降低的问题。

删除的方法也很简单,因为好友的每一条信息的键都是接收到该信息时刻的时间戳,这也是前面这样设计的原因:

这个字典则是以接收到消息那一刻的时间戳为键,以该时间戳对应的信息内容为值。这样设计便于后面找到撤回的消息。

import copy
def del_overdue_msg(cur_timestamp):
log_copy = copy.deepcopy(log)
for friend in log_copy:
for timestamp in log_copy[friend]:
if cur_timestamp - timestamp > 2.01 * 60:
log[friend].pop(timestamp)

因为不能在循环字典的时候对其进行更改,所以先copy一份,然后处理,注意要引入copy包。代码中设计的时间间隔为2.01分钟,是为了留出一定的Buffer,避免因程序运行原因导致错过什么消息。为了简化代码,把这段代码拿出来写到一个独立的方法中,然后在每次接收到消息时调用它。

@itchat.msg_register([TEXT, MAP, CARD, NOTE, SHARING])
def text_reply(msg):
cur_timestamp = time.time()
del_overdue_msg(cur_timestamp)
...

生成撤回信息

当好友撤回已经发送的消息时,也会进入我们的程序,就像前面控制台的截图:

ide撤回 python python怎样撤回_信息内容

此时程序判断的消息发送者就是撤回消息的好友昵称,而消息的格式则是固定的:

msg_pattern = '"{}" 撤回了一条消息'.format(sender_name)

如果接收到的消息内容符合这个格式,就能确定某个好友撤回了消息(不考虑好友故意发送同样的信息的情况,这个真处理不了)。

但是可能该好友短时间连续发了多条消息,怎么确定是哪一条呢?目前我了解到的,有多条临时存储的消息的情况是不能唯一确定被撤回的是哪一条信息的。不过从逻辑上讲,一般被撤回的消息大概率都是最新的一条。在不确定的情况下,可以先显示最新的一条,其余消息在后面作为备选。

还有一个要注意的点。好友有可能会连续撤回好几条消息,所以程序中已经作为撤回消息返回的部分还不能立即从内存中删除,避免找不到真正撤回的消息。在整个处理过程中,临时存储的消息只会在前面说的“2分钟”的逻辑下才会被删除。

有了策略,就可以开始撸代码了:

target_msg_pattern = '"{}" 撤回了一条消息'.format(sender_name)
if content == target_msg_pattern:
return_msg = '【{}】撤回了一条消息:\n'.format(sender_name)
if len(log[sender_name].items()) == 0:
return_msg = '缓存信息列表为空!'
else:
return_msg += log[sender_name].items()[-1][-1] + '\n'
if len(log[sender_name].items()) > 1:
msgs = [msg for timestamp, msg in log[sender_name].items()[:-1]]
return_msg += '也有可能是下列信息中的某一条:\n' + '\n'.join(msgs)

下面是实际效果:

机器人“小帮帮”发过来一条消息:

ide撤回 python python怎样撤回_ide撤回 python_02

控制台给出输出:

ide撤回 python python怎样撤回_github_03

机器人撤回消息:

ide撤回 python python怎样撤回_信息内容_04

控制台给出撤回的信息内容:

ide撤回 python python怎样撤回_python打错了怎么撤回_05

如果是连续的多条消息,则是这样的:

收到信息

ide撤回 python python怎样撤回_ide撤回 python_06

控制台显示收到

ide撤回 python python怎样撤回_信息内容_07

撤回第一条信息

ide撤回 python python怎样撤回_ide撤回 python_08

控制台显示可能撤回的信息

ide撤回 python python怎样撤回_github_09

撤回第二条信息

ide撤回 python python怎样撤回_时间戳_10

控制台显示可能撤回的信息

ide撤回 python python怎样撤回_ide撤回 python_11

大概就是这样。

下期预告

本来是打算今天写完的,但奈何这一期太长,要看完可能是个体力活。所以,下一期我们会实现将整理好的撤回消息文本发送给“文件传输助手”,以便我们及时知道又有谁撤回了什么见不得人的消息。

后记

不管写什么,希望能跟更多人沟通,有问题或者需求随时欢迎交流。

我所有的项目源码都会放在下面的github仓库里面,有需要可以参考,有问题欢迎指正,谢谢!

本文代码:

https://github.com/TitusWongCN/AntiInfoWithdrawal

所有代码:

https://github.com/TitusWongCN/