官方華麗的術(shù)語---略
一個(gè)自動(dòng)回復(fù)機(jī)器人
微信自上線以來,一直沒有自動(dòng)回復(fù)的功能端壳,想必是有他們的理念。但是有些人群损谦,確實(shí)對(duì)此功能有一定需求,我舉兩個(gè)栗子:
不愿時(shí)刻被消息打擾的人
消息需要批量處理的人們(比如微商)
設(shè)計(jì)了幾個(gè)功能:
[x] 收到消息立即自動(dòng)回復(fù)
[x] 收到消息延遲指定時(shí)間回復(fù)
[x] 對(duì)不同好友定制不同的回復(fù)內(nèi)容
[x] 在手機(jī)端隨時(shí)進(jìn)行控制
itchat的原理是通過掃碼進(jìn)行遠(yuǎn)程微信網(wǎng)頁端的登錄照捡,然后通過在移動(dòng)端進(jìn)行操作,網(wǎng)頁端進(jìn)行響應(yīng)栗精,來實(shí)現(xiàn)一些功能,雖然似乎繞了個(gè)彎子,但是在微信的限制下鹿寨,這似乎已經(jīng)是一種最佳的方式了,猶如戴著鐐銬起舞脚草。
我們可以先通過設(shè)定幾個(gè)全局變量來做作為功能的開關(guān)和保存數(shù)據(jù)的容器。
#自動(dòng)回復(fù)開關(guān)
SWITCH_REPLY=True
#延遲回復(fù)開關(guān)
SWITCH_DELAY=False
#延遲時(shí)間
DELAY_TIME=120
#消息前綴開關(guān)
SWITCH_PREFIX=True
#消息前綴內(nèi)容
PREFIX_CONTENT="[自動(dòng)回復(fù)]"
#回復(fù)內(nèi)容字典
REPLY_DICT={}
#延遲回復(fù)字典
DELAY_REPLY_DICT={}
然后通過判斷web端在”文件管理器“中接收到的字符串指令來進(jìn)行不同操作馏慨,這一部分的代碼比較簡單且冗長,這里就不貼出來了写隶,完整源碼地址將會(huì)在文末給出。
假如此時(shí)我們收到了朋友的消息樟澜,需要程序給出自動(dòng)回復(fù)。
#獲取發(fā)送消息的朋友的信息
target_friend=itchat.search_friends(userName = msg['FromUserName'])
if target_friend:
#獲取ta的昵稱
nickName=target_friend['NickName']
if not REPLY_DICT.__contains__(nickName):
#設(shè)置默認(rèn)回復(fù)
REPLY_DICT[nickName]="抱歉我有事暫未看到消息秩贰,稍后回復(fù),若有急事可以電話聯(lián)系(?ω?`)"
reply_content=REPLY_DICT[nickName]
#判斷自動(dòng)回復(fù)開關(guān)
if SWITCH_REPLY:
#判斷延時(shí)回復(fù)開關(guān)
if SWITCH_DELAY:
localtime = time.time()
DELAY_REPLY_DICT[nickName]=[localtime,msg['FromUserName']]
print (DELAY_REPLY_DICT)
if not SWITCH_DELAY:
#判斷消息前綴開關(guān)
if SWITCH_PREFIX:
reply_content = PREFIX_CONTENT + REPLY_DICT[nickName]
else:
reply_content = REPLY_DICT[nickName]
#發(fā)送消息
itchat.send(reply_content, toUserName=msg['FromUserName'])
收到朋友消息即時(shí)進(jìn)行自動(dòng)回復(fù)是很簡單的毒费,但是如何去做延時(shí)發(fā)送回復(fù)消息呢愈魏?(至于做這個(gè)功能有沒有必要的問題可以先擱置觅玻,不過我認(rèn)為在很多場(chǎng)景下是需要這個(gè)功能的培漏,大家也可以在評(píng)論區(qū)討論在什么場(chǎng)景下需要延遲自動(dòng)回復(fù))現(xiàn)在就回到技術(shù)的問題,如何實(shí)現(xiàn)可設(shè)置時(shí)間的延時(shí)自動(dòng)回復(fù)牌柄。
我先談一談我的想法,拋磚引玉珊佣。一般發(fā)送消息需要用到隊(duì)列蹋宦,進(jìn)行入隊(duì)和出隊(duì)咒锻,我在這里設(shè)置了一個(gè)字典來保存消息發(fā)送者的數(shù)據(jù)。鍵為消息發(fā)送者的昵稱惑艇,值是一個(gè)長度為2的數(shù)組,分別保存消息發(fā)送者的微信id和接收消息時(shí)的時(shí)間戳。這樣我將每條發(fā)送過來的朋友信息保存在這個(gè)字典中须板,再通過將設(shè)定延遲時(shí)間同消息時(shí)間戳求和與當(dāng)前時(shí)間戳進(jìn)行對(duì)比,若當(dāng)前時(shí)間戳較大习瑰,那么執(zhí)行發(fā)送消息的操作秽荤。
此時(shí)再開啟一個(gè)線程作為定時(shí)任務(wù),定時(shí)去檢測(cè)字典中每條數(shù)據(jù)是否到達(dá)了發(fā)送的臨界要求(當(dāng)前時(shí)間戳>=消息時(shí)間戳+設(shè)定的延遲時(shí)間)Python中有個(gè)專門做定時(shí)任務(wù)的模塊叫sched窃款,但是我嘗試了一下,sched會(huì)阻塞當(dāng)前主線程晨继,也會(huì)阻塞itchat的線程,所以并不合適紊扬。這里我還是采用了threading的Timer來充當(dāng)定時(shí)器,不過要注意使用遞歸餐屎,否則將會(huì)出現(xiàn)運(yùn)行一次就結(jié)束的情況。
#延遲發(fā)送消息的函數(shù)
def delay_reply():
#print("開始執(zhí)行")
global DELAY_REPLY_DICT
if SWITCH_DELAY:
while len(DELAY_REPLY_DICT)>0:
localtime = time.time()
# print (localtime)
# print (DELAY_REPLY_DICT[item][0])
# print (int(DELAY_TIME))
for item in list(DELAY_REPLY_DICT.keys()):
if SWITCH_REPLY:
reply_content = item + "," + str(round(int(DELAY_TIME) / 60, 1)) + "分鐘過去了腹缩," + REPLY_DICT[item]
itchat.send(reply_content, toUserName=DELAY_REPLY_DICT[item][1])
# print ("發(fā)送消息")
del DELAY_REPLY_DICT[item]
print (DELAY_REPLY_DICT)
global timer1
timer1=threading.Timer(DELAY_TIME,delay_reply)
timer1.start()
到此為止藏鹊,主要的功能已經(jīng)實(shí)現(xiàn)了盘寡,我用一個(gè)測(cè)試賬號(hào)對(duì)我的微信進(jìn)行了各種測(cè)試菇曲,看一下以下截圖:
這時(shí)功能基本已經(jīng)完成了萧朝,這就結(jié)束了嗎竖配?別著急,再想想有沒有需要完善一下的地方盯漂?用過微信web端的同學(xué)應(yīng)該知道,當(dāng)web端長期處于未操作的狀態(tài)下會(huì)失去連接羞延。在我們這個(gè)情況下嗤谚,假如你長時(shí)間未收到微信消息,后臺(tái)程序?qū)?huì)與微信失去連接籍胯,再次開啟需要登上服務(wù)器重啟程序,這顯然非常麻煩。有沒有什么簡單的解決辦法呢?我想到一些應(yīng)用的后臺(tái)通常會(huì)做一道心跳檢測(cè)機(jī)制,那我就模仿這個(gè)思路炸裆,定時(shí)給我的”文件管理器“發(fā)一個(gè)字符串,來保持連接烹看。
def keep_alive():
text="保持登錄"
itchat.send(text, toUserName="filehelper")
global timer2
timer2 = threading.Timer(60*60,keep_alive)
timer2.start()
最后,我們需要將這個(gè)程序發(fā)布在服務(wù)器上惯殊,讓它全天候?yàn)槲业奈⑿欧?wù)。這里需要注意土思,如果僅用python xxxx.py
來運(yùn)行的話,關(guān)閉shell會(huì)導(dǎo)致進(jìn)程結(jié)束己儒,所以我們需要使用nohup python xxxx.py &
來全方位守護(hù)進(jìn)程,這里啰嗦一句闪湾,nohup和&的功能是不一樣的,很多人容易混淆途样,感興趣的話可以去查下資料區(qū)分一下。
到此何暇,微信經(jīng)過了我們的稍稍調(diào)教,已經(jīng)乖了一點(diǎn)赖晶。但是辐烂,這遠(yuǎn)遠(yuǎn)不夠,思路可以繼續(xù)拓展纠修,比如實(shí)現(xiàn)通過手機(jī)微信來控制電腦的開關(guān)機(jī),電腦軟件的啟動(dòng)關(guān)閉扣草,都是可以的,甚至實(shí)現(xiàn)控制家中的空調(diào)颜屠,理論上也未嘗不可,只是比較有難度甫窟。
簡單分析微信好友信息
上文提到,既然我們能通過itchat來獲取好友的信息粗井,name自然會(huì)有很多好玩的信息(這里不做具體解析)。
性別比例
def get_sex():
# 獲取好友數(shù)據(jù)
my_friends = itchat.get_friends(update=True)[0:]
sex = {"male": 0, "female": 0, "other": 0}
for item in my_friends[1:]:
s = item["Sex"]
if s == 1:
sex["male"] += 1
elif s == 2:
sex["female"] += 1
else:
sex["other"] += 1
total = len(my_friends[1:])
# 開始畫餅圖
attr = list(sex.keys())
v1 = list(sex.values())
pie = Pie("好友性別比例")
pie.add("", attr, v1, v1, is_label_show=True)
pie.render(path="sex_html/sex.html")
全國好友省級(jí)分布
def friends_province():
# 獲取好友省份
province= get_data("Province")
# 分類
province_distribution = {}
for item in province:
#刪除英文省份懒构,因?yàn)橹袊貓D表中沒有
if bool(re.search('[a-z]',item)):
continue
elif not province_distribution.__contains__(item):
province_distribution[item] = 1
else:
province_distribution[item] += 1
#將省份名為空的刪除
province_distribution.pop('')
#提取地圖接口需要的數(shù)據(jù)格式
# print(province_distribution)
province_keys=province_distribution.keys()
province_values=province_distribution.values()
return province_keys,province_values
注意:這里需要自己安裝地圖耘擂,否則會(huì)只顯示南海諸島問題胆剧,http://www.reibang.com/p/20fd061d0b96:
解決方案醉冤,手動(dòng)安裝地圖
- 全球國家地圖: echarts-countries-pypkg (1.9MB): 世界地圖和 213 個(gè)國家秩霍,包括中國地圖
- 中國省級(jí)地圖: echarts-china-provinces-pypkg (730KB):23 個(gè)省冤灾,5 個(gè)自治區(qū)
- 中國市級(jí)地圖: echarts-china-cities-pypkg (3.8MB):370 個(gè)中國城市
需要這些地圖的朋友,可以裝 pip 命令行:
pip install echarts-countries-pypkg
pip install echarts-china-provinces-pypkg
pip install echarts-china-cities-pypkg
特別注明韵吨,中國地圖在 echarts-countries-pypkg 里。
好友標(biāo)簽
def friends_signature():
signature = get_data("Signature")
wash_signature=[]
for item in signature:
#去除emoji表情等非文字
if "emoji" in item:
continue
rep = re.compile("1f\d+\w*|[<>/=【】『』♂ω]")
item=rep.sub("", item)
wash_signature.append(item)
words="".join(wash_signature)
print(wash_signature)
wordlist = jieba.cut(words, cut_all=True)
word_space_split = " ".join(wordlist)
global NickName
global Sex
# print(NickName, Sex)
# 圖片的作用:生成的圖片是這個(gè)圖片的兩倍大小
# 根據(jù)性別選擇對(duì)應(yīng)的性別模板圖
if Sex == 2:
coloring = np.array(Image.open("standard/girl.jpg"))
elif Sex == 1:
coloring = np.array(Image.open("standard/boy.jpg"))
else:
coloring = np.array(Image.open("standard/num.jpg"))
# simkai.ttf 必填項(xiàng) 識(shí)別中文的字體椿疗,例:simkai.ttf,
my_wordcloud = WordCloud(background_color="white", max_words=800,
mask=coloring, max_font_size=120, random_state=30, scale=2,font_path="fonts/STKAITI.TTF").generate(word_space_split)
image_colors = ImageColorGenerator(coloring)
plt.imshow(my_wordcloud.recolor(color_func=image_colors))
plt.imshow(my_wordcloud)
plt.axis("off")
plt.show()
# 保存圖片
my_wordcloud.to_file('Signature/signature.png')
說明兩點(diǎn):
1届榄、一定要有標(biāo)簽樣版圖,生成對(duì)應(yīng)樣式
2铝条、要添加識(shí)別中文的字體
Github:https://github.com/silencesmile/Young_WeChart
如果您喜歡靖苇,希望能給個(gè)小星星