python實(shí)現(xiàn)12306搶票及自動(dòng)郵件發(fā)送提醒付款功能
#寫在前面,這個(gè)程序我已經(jīng)弄出來了,但是因?yàn)辄S牛泛濫以及懶人太多,整個(gè)程序的代碼就不貼出來了,這里純粹就是技術(shù)交流。
只做技術(shù)交流、、、、、
嗯,程序結(jié)束后,自己還是得手動(dòng)付款。
廢話不多說,下面就直接開始技術(shù)主要部分闡述。
先講理論部分:首先我們需要代碼實(shí)現(xiàn)一個(gè)瀏覽器功能,那么模塊基本上可以確定urllib.parse、urllib.request,這兩個(gè)包都是和網(wǎng)址有關(guān)的模塊,那么咱們?nèi)サ卿浺粋€(gè)網(wǎng)址,特別是有驗(yàn)證碼這些的網(wǎng)址,我們登錄進(jìn)去是不是就行了?答案是對(duì)的,但是我們用代碼實(shí)現(xiàn)的話,這個(gè)網(wǎng)址可能每次都有可能被代碼去請(qǐng)求,那么服務(wù)器怎么知道我們是一個(gè)人,而不是多個(gè)瀏覽器不同的用戶呢?
此時(shí)cookie就非常重要了,在代碼中設(shè)置好cookie,那么對(duì)方服務(wù)器自然就知道我們是一個(gè)人,比較服務(wù)器都是這么區(qū)分的。python3中 cookie這個(gè)功能是封裝在http.cookiejar這個(gè)模塊之內(nèi)。好了,代碼如下:
# coding=utf-8 # author: Jason # time:2018/1/16 20:00:00 #version:1.0 import urllib.request as ul import urllib.parse as uz import http.cookiejar as cookielib from json import loads c=cookielib.LWPCookieJar()#先把cookie對(duì)象存儲(chǔ)為cookiejar的對(duì)象 cookie = ul.HTTPCookieProcessor(c)#把cookiejar對(duì)象轉(zhuǎn)換為一個(gè)handle opener = ul.build_opener(cookie)#建立一個(gè)模擬瀏覽器,需要handle作為參數(shù) ul.install_opener(opener)#安裝一個(gè)全局模擬瀏覽器,代表無論怎么訪問都是一個(gè)瀏覽器操作而不是分開獲取驗(yàn)證碼等msg
好了,如此一來,我們代碼的初步實(shí)現(xiàn)已經(jīng)完成,接下來就是進(jìn)入網(wǎng)絡(luò)分析部分
首先可以使用google瀏覽器或者搜狗瀏覽器(本人用的搜狗),打開F12,也就是開發(fā)者模式,登錄12306的登錄地址 https://kyfw.12306.cn/otn/login/init
兩個(gè)紅圈中第二個(gè)是驗(yàn)證碼來源,此時(shí)我們只需要記錄這個(gè)網(wǎng)頁(點(diǎn)進(jìn)去)的詳細(xì)情況,寫入代碼當(dāng)中,python3中urllib.request這個(gè)模塊打開既可
如此便是驗(yàn)證碼來源,那么如何用代碼捕捉呢?首先我們可以先亂輸入密碼,亂點(diǎn)驗(yàn)證碼,然后我們直接點(diǎn)擊登錄
多了一個(gè)很奇妙的東西,此時(shí),這里就是驗(yàn)證碼驗(yàn)證的網(wǎng)址,那么我們是不是應(yīng)該記錄下來呢?很簡(jiǎn)單,到Headers里面就全都看得到了
上面那個(gè)是服務(wù)器驗(yàn)證網(wǎng)址,下面就是我們回復(fù)給他的東西,那么那個(gè)163,121其實(shí)就是我亂點(diǎn)的驗(yàn)證碼坐標(biāo)了。至于為啥是坐標(biāo),因?yàn)樗怯檬髽?biāo)去點(diǎn)圖片,那么他只可能是記錄坐標(biāo),除非他自己搞了一套人工智能驗(yàn)證圖片,但基于他幾年前就這么玩了,人工智能根本沒有怎么開始,他自然只能是最原始的技術(shù)而已。
那么這代表了他是先驗(yàn)證驗(yàn)證碼,那么驗(yàn)證密碼的在哪?自然是需要驗(yàn)證碼這關(guān)能過,那我們輸一個(gè)正確的驗(yàn)證碼,再寫個(gè)錯(cuò)密碼,登錄
此時(shí)可以看到,和驗(yàn)證碼一樣的方法,我們的回復(fù)與驗(yàn)證都在這幾個(gè)圈了,還記得上面驗(yàn)證碼失敗的時(shí)候回復(fù)給我們的code是不是有個(gè)數(shù)字?這個(gè)也很重要,那么可以看看我們的驗(yàn)證成功的驗(yàn)證碼返回給我們的是什么東西
這次我們看到了,驗(yàn)證碼成功,顯示是4,好,那我們不就可以進(jìn)行條件判斷了么?
那么如何打開一個(gè)網(wǎng)址然后把我們點(diǎn)的東西一起發(fā)過去呢?上代碼
headers={'User-Agent':'Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/63.0.3239.84 Safari/537.36'}#先寫個(gè)頭,表示我這是瀏覽器用戶登錄而不是代碼登錄,如果不寫,代碼默認(rèn)用的簽名之類的是編程語言的標(biāo)識(shí),這樣對(duì)方服務(wù)器很容易就發(fā)現(xiàn)你是個(gè)腳本了
def get_code():#獲取驗(yàn)證碼的步驟 req = ul.Request('https://kyfw.12306.cn/passport/captcha/captcha-image?login_site=E&module=login&rand=sjrand&0.6758635422370105') req.headers = headers code_file = opener.open(req).read()#此時(shí)為瀏覽器的open而不再是ul.urlopen,下同 with open(r'C:\Users\Administrator\Desktop\12306自動(dòng)搶票\code.png','wb')as f: f.write(code_file)
把驗(yàn)證碼直接下載后方電腦上,后面要坐標(biāo)只需要打開這個(gè)圖既可輸入,坐標(biāo)的輸入方式我用字典表示給大家看{1:(45,45)}{2:(120,45)}{3:(180,45)}{4:(255,45)}{5:(45,120)}{6:(120,120)}{7:(180,120)}{8:(255,120)}
根據(jù)這個(gè)驗(yàn)證碼的排序,我相信讀者應(yīng)該知道順序怎么來的吧,比較坐標(biāo)就能懂了。
繼續(xù)
def main_(): get_code() code = input('輸入驗(yàn)證碼:') req = ul.Request('https://kyfw.12306.cn/passport/captcha/captcha-check') req.headers = headers data = { 'answer':code, 'login_site':'E', 'rand':'sjrand' } data = uz.urlencode(data).encode()#把字典轉(zhuǎn)換為URL query string,此時(shí)是str,要把它變?yōu)閎yts。 html = opener.open(req,data= data).read().decode()#讀取出來是byts格式,轉(zhuǎn)換為‘utf-8(默認(rèn)) print(html) result = loads(html) if result['result_code']=='4': print('驗(yàn)證碼通過') rep = ul.Request('https://kyfw.12306.cn/passport/web/login') rep.headers = headers data = {'username':'這里就是你用戶名', 'password':'這里就是你的密碼', 'appid':'otn' } data = uz.urlencode(data).encode() #看到了嗎,這就是你給服務(wù)器回復(fù)的東西 html1 = opener.open(rep,data = data ).read().decode() result1 = loads(html1) if result1['result_code'] == 0: print('賬戶密碼驗(yàn)證通過') else: print(result1['result_message']) else: print('驗(yàn)證碼校驗(yàn)失敗,重來') if __name__ == '__main__': main_()
此時(shí),咱們就過了驗(yàn)證碼密碼這一關(guān),后面是不是又要查票?那么同樣的方法,我們就可以以此類推到最后一步,這里就不一一貼代碼了
ps:查代碼這幾步的信息可是很重要喔,我們要把它記錄好,并且這里面的信息包含了各種作為信息以及他們的順序,多測(cè)試幾次基本都能搞出來,這里就是提醒一點(diǎn)
找找規(guī)律,然后用split的方法完全就可以切割出來,然后一個(gè)循環(huán),就可以得到第幾個(gè)元素是我們要的,那么后面就可以標(biāo)志判斷返回值如果是無,就沒票可以繼續(xù)查詢,直到有票就可以下一步;
那么有票的話,后面一樣也是以此類推的方式,代碼我就不重現(xiàn)了,很簡(jiǎn)單,我就把后面出現(xiàn)需要請(qǐng)求的網(wǎng)址都發(fā)出來供大家觀摩
查詢車票信息
url = 'https://kyfw.12306.cn/otn/leftTicket/queryZ?leftTicketDTO.train_date=%s&leftTicketDTO.from_station=%s&leftTicketDTO.to_station=%s&purpose_codes=ADULT'%(train_data,from_station,to_station)
req = ul.Request('https://kyfw.12306.cn/otn/leftTicket/submitOrderRequest')#確定訂單信息 req = ul.Request("https://kyfw.12306.cn/otn/confirmPassenger/initDc")#驗(yàn)證訂單 req = ul.Request('https://kyfw.12306.cn/otn/confirmPassenger/getPassengerDTOs')#準(zhǔn)備跨到下單中的過度 req = ul.Request('https://kyfw.12306.cn/otn/confirmPassenger/checkOrderInfo')#檢查訂單信息 req = ul.Request('https://kyfw.12306.cn/otn/confirmPassenger/getQueueCount')#信息提交給服務(wù)器,準(zhǔn)備進(jìn)入下單步驟 req = ul.Request('https://kyfw.12306.cn/otn/confirmPassenger/confirmSingleForQueue')#正式進(jìn)入下單步驟 req = ul.Request('https://kyfw.12306.cn/otn/confirmPassenger/queryOrderWaitTime?random=%s&tourFlag=dc&_json_att=&REPEAT_SUBMIT_TOKEN=%s'%(numb,time.time()))#下單確認(rèn)中,此時(shí)這個(gè)網(wǎng)址一般是進(jìn)行兩次訪問,不知為何,我還是做了兩次訪問,numb是前面查詢車票點(diǎn)擊預(yù)定回復(fù)我們的信息中的一條,有點(diǎn)難找喔,我曾經(jīng)找了三天。。。當(dāng)然是因?yàn)樽约翰蛔屑?xì)而已 zreq = ul.Request("https://kyfw.12306.cn/otn/confirmPassenger/resultOrderForDcQueue")#最后的結(jié)果回執(zhí),如果一切都順利,那么票就已經(jīng)訂了。我一般是打印他返回的內(nèi)容 ''' zreq = ul.Request("https://kyfw.12306.cn/otn/confirmPassenger/resultOrderForDcQueue") zreq.headers = headers data ={"REPEAT_SUBMIT_TOKEN":"%s"%numb, "_json_att": "", "orderSequence_no":orderId } data = uz.urlencode(data).encode() html = opener.open(zreq,data=data).read().decode() result = loads(html) print('代碼全部過完,回去登錄下是否搞定') print(result) print(result['data']['submitStatus']) if result['data']['submitStatus'] == True: print('購(gòu)票成功') return True else: print('購(gòu)票失敗,重試其他列車') continue ''' 最終的回執(zhí)代碼詳細(xì) 信息,讀者可以自己嘗試多次,得到自己的回復(fù)代碼確認(rèn)是否購(gòu)票成功,因?yàn)閞esult['data']['submitStatus']==True只不過是確認(rèn)訂單狀態(tài)而已,這個(gè)被我改動(dòng)過,你可以多次嘗試
最后的最后,火車票預(yù)訂成功只有30分鐘支付時(shí)間,所以我為了防止訂好票但是我人不在,特意寫了qq郵件通知
qq郵件通知:
def email():#這是我訂票后給自己發(fā)郵件的函數(shù) import smtplib from email.mime.text import MIMEText import time text = '已經(jīng)為%s搶到票,速度登錄12306付款,用戶名:%s,密碼:%s'%(NAME,username,password) msg = MIMEText(text, 'plain', 'utf-8') msg_From = '2059****16@qq.com' msg_To = '5043****75@qq.com'#是的,我有兩個(gè)qq,一個(gè)發(fā)一個(gè)收 smtpSever = 'smtp.qq.com' # qq郵箱的smtp Sever地址 smtpPort = '465' # 開放的端口 sqm = 'q********eghe' # 在登錄smtp時(shí)需要login中的密碼應(yīng)當(dāng)使用授權(quán)碼而非賬戶密碼 msg['from'] = msg_From msg['to'] = msg_To msg['subject'] = 'Python自動(dòng)郵件-%s' % time.ctime() smtp = smtplib smtp = smtplib.SMTP_SSL() ''' smtplib的connect(連接到郵件服務(wù)器)、login(登陸驗(yàn)證)、sendmail(發(fā)送郵件) ''' smtp.connect(smtpSever, smtpPort) smtp.login(msg_From, sqm) smtp.sendmail(msg_From, msg_To, str(msg)) # s = smtplib.SMTP("localhost") # s.send_message(msg) smtp.quit() print('郵件已發(fā)送~你可以安心去玩了') def emailforcode():#此函數(shù)是防止查詢有票但12306賬號(hào)掉線人不在無法訂票的提醒 import smtplib from email.mime.text import MIMEText import time text = '%s賬號(hào)下線,速度登錄驗(yàn)證12306' % NAME msg = MIMEText(text, 'plain', 'utf-8') msg_From = '205****516@qq.com' msg_To = '50****75@qq.com' smtpSever = 'smtp.qq.com' # qq郵箱的smtp Sever地址 smtpPort = '465' # 開放的端口 sqm = 'qowa*******ghe' # 在登錄smtp時(shí)需要login中的密碼應(yīng)當(dāng)使用授權(quán)碼而非賬戶密碼 msg['from'] = msg_From msg['to'] = msg_To msg['subject'] = 'Python自動(dòng)郵件-%s' % time.ctime() smtp = smtplib smtp = smtplib.SMTP_SSL() ''' smtplib的connect(連接到郵件服務(wù)器)、login(登陸驗(yàn)證)、sendmail(發(fā)送郵件) ''' smtp.connect(smtpSever, smtpPort) smtp.login(msg_From, sqm) smtp.sendmail(msg_From, msg_To, str(msg)) # s = smtplib.SMTP("localhost") # s.send_message(msg) smtp.quit() print('郵件已發(fā)送~')
如此就大功告成了。
不能發(fā)完整的代碼(本身目的就是為了技術(shù)交流而已,防止懶人盜碼亂搞),但是我相信各位開發(fā)中的朋友們只要有邏輯,有開頭,只要自己肯動(dòng)手,都可以自己鉆研出來,舉一反三。畢竟我就是這樣搞出來的,我從來都相信,只要肯學(xué),都會(huì)學(xué)會(huì),只要肯做,都可以做成。
總結(jié)
以上所述是小編給大家介紹的python實(shí)現(xiàn)12306搶票及自動(dòng)郵件發(fā)送提醒付款功能,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
python正則表達(dá)式之對(duì)號(hào)入座篇
正則表達(dá)式是對(duì)字符串操作的一種邏輯公式,就是用事先定義好的一些特定字符、及這些特定字符的組合,組成一個(gè)“規(guī)則字符串”,這個(gè)“規(guī)則字符串”用來表達(dá)對(duì)字符串的一種過濾邏輯2018-07-07Python3 字典dictionary入門基礎(chǔ)附實(shí)例
Python字典是另一種可變?nèi)萜髂P?,且可存?chǔ)任意類型對(duì)象,如字符串、數(shù)字、元組等其他容器模型,字典由鍵和對(duì)應(yīng)值成對(duì)組成,字典也被稱作關(guān)聯(lián)數(shù)組或哈希表2020-02-02如何使用python的subprocess執(zhí)行命令、交互、等待、是否結(jié)束及解析JSON結(jié)果
這篇文章主要給大家介紹了關(guān)于如何使用python的subprocess執(zhí)行命令、交互、等待、是否結(jié)束及解析JSON結(jié)果的相關(guān)資料,subprocess模塊提供了一種簡(jiǎn)單的方法來創(chuàng)建和管理子進(jìn)程,它可以讓我們?cè)赑ython程序中執(zhí)行外部命令,獲取命令的輸出和錯(cuò)誤信息,需要的朋友可以參考下2023-12-12利用Python實(shí)現(xiàn)原創(chuàng)工具的Logo與Help
這篇文章主要給大家介紹了關(guān)于如何利用Python實(shí)現(xiàn)原創(chuàng)工具的Logo與Help的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考借鑒,下面來一起看看吧2018-12-12python opencv minAreaRect 生成最小外接矩形的方法
這篇文章主要介紹了python opencv minAreaRect 生成最小外接矩形的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2019-07-07python統(tǒng)計(jì)文本字符串里單詞出現(xiàn)頻率的方法
這篇文章主要介紹了python統(tǒng)計(jì)文本字符串里單詞出現(xiàn)頻率的方法,涉及Python字符串操作的相關(guān)技巧,需要的朋友可以參考下2015-05-05