年底搶車票必看!看完之後再也不用擔心車票問題!
臨近過年,又到了一年一度的搶車票大賽,搶不到車票的過年又不能好好地回一趟家。
由於最近在學Python,所以用Python寫了這個12306搶票指令碼,分享出來,希望幫助更多的人過年能夠回去跟家人團聚,然後與大家共同交流和學習,有不對的地方,請大家多多指正。話不多說,進入正題:
在進入正題之前,我想說明一下,由於12306官網的改版更新,所以指令碼作了一點小小的變化,具體修改後的原始碼,可以來小編的QQ裙領取(609616831)有不會操作的小夥伴也能來哦,這邊有具體操作的方法。
這個指令碼目前只能刷一趟車的,人數可以是多個,支援選取作為型別等。
實現思路是splinter.browser模擬瀏覽器登陸和操作,由於12306的驗證碼不好自動識別,所以,驗證碼需要使用者進行手動識別,並進行登陸操作,之後的事情,就交由指令碼來操作就可以了,下面是我測試時候的一些截圖:
第一步:如下圖,首先輸入搶票基本資訊
第二步:然後進入登入頁,需要手動輸入驗證碼,並點選登陸操作
第三步:登陸後,自動進入到搶票頁面,如下圖這樣的
最後:就是坐等刷票結果就好了,如下圖這樣,就說是刷票成功了,刷到票後,會進行簡訊和郵件的通知,請記得及時前往12306進行支付,不然就白搶了。
Python執行環境:python3.6
用到的模組:re、splinter、time、sys、httplib2、urllib、smtplib、email
未安裝的模組,請使用pip instatll進行安裝,例如:pip install splinter
如下程式碼是這個指令碼所有用到的模組引入:
-
import re from splinter.browser import Browser from time import sleep import sys import httplib2 from urllib import parse import smtplib from email.mime.text import MIMEText
刷票前資訊準備,我主要說一下始發站和目的地的cookie值獲取,因為輸入城市的時候,需要通過cookie值,cookie值可以通過12306官網,然後在F12(相信所有的coder都知道這個吧)的network裡面的查詢請求cookie中可以看到,在請求的header裡面可以找到,_jc_save_fromStation值是出發站的cookie,_jc_save_toStation的值是目的地的cookie,然後加入到程式碼裡的城市的cookie字典city_list裡即可,鍵是城市的首字母,值是cookie值的形式。
搶票,肯定需要先登入,我這裡模擬的登入操作,會自動填充12306的賬號名和密碼,當然,你也可以在開啟的瀏覽器中修改賬號和密碼,實現的關鍵程式碼如下:
-
def do_login(self): """登入功能實現,手動識別驗證碼進行登入""" self.driver.visit(self.login_url) sleep(1) self.driver.fill('loginUserDTO.user_name', self.user_name) self.driver.fill('userDTO.password', self.password) print('請輸入驗證碼……') while True: if self.driver.url != self.init_my_url: sleep(1) else: break 登入之後,就是控制刷票的各種操作處理了,這裡,我就不貼程式碼了,因為程式碼比較多,別擔心,在最後,我會貼出完整的程式碼的。 當刷票成功後,我會進行簡訊和郵件的雙重通知,當然,這裡簡訊通知的平臺,就看你用那個具體來修改程式碼了,我用的是互億無線的體驗版的免費簡訊通知介面;傳送郵件模組我用的是smtplib,傳送郵件伺服器用的是163郵箱,如果用163郵箱的話,你還沒有設定客戶端授權密碼,記得先設定客戶端授權密碼就好了,挺方便的。以下是主要實現程式碼: def send_sms(self, mobile, sms_info): """傳送手機通知簡訊,用的是-互億無線-的測試簡訊""" host = "106.ihuyi.com" sms_send_uri = "/webservice/sms.php?method=Submit" account = "C59782899" pass_word = "19d4d9c0796532c7328e8b82e2812655" params = parse.urlencode( {'account': account, 'password': pass_word, 'content': sms_info, 'mobile': mobile, 'format': 'json'} ) headers = {"Content-type": "application/x-www-form-urlencoded", "Accept": "text/plain"} conn = httplib2.HTTPConnectionWithTimeout(host, port=80, timeout=30) conn.request("POST", sms_send_uri, params, headers) response = conn.getresponse() response_str = response.read() conn.close() return response_str def send_mail(self, receiver_address, content): """傳送郵件通知""" # 連線郵箱伺服器資訊 host = 'smtp.163.com' port = 25 sender = '[email protected]' # 你的發件郵箱號碼 pwd = '******' # 不是登陸密碼,是客戶端授權密碼 # 發件資訊 receiver = receiver_address body = '<h2>溫馨提醒:</h2><p>' + content + '</p>' msg = MIMEText(body, 'html', _charset="utf-8") msg['subject'] = '搶票成功通知!' msg['from'] = sender msg['to'] = receiver s = smtplib.SMTP(host, port) # 開始登陸郵箱,併發送郵件 s.login(sender, pwd) s.sendmail(sender, receiver, msg.as_string()) 說了那麼多,感覺都是說了好多廢話啊,哈哈,不好意思,耽誤大家時間來看我瞎扯了,我貼上大家最關心的原始碼,請接碼,大家在嘗試執行過程中,有任何問題,可以給我留言或者私信我,我看到都會及時回覆大家的: #!/usr/bin/env python # -*- coding: utf-8 -*- """ 通過splinter刷12306火車票 可以自動填充賬號密碼,同時,在登入時,也可以修改賬號密碼 然後手動識別驗證碼,並登陸,接下來的事情,交由指令碼來做了,靜靜的等待搶票結果就好(刷票過程中,瀏覽器不可關閉) author: cuizy time: 2018-05-30 """ import re from splinter.browser import Browser from time import sleep import sys import httplib2 from urllib import parse import smtplib from email.mime.text import MIMEText class BrushTicket(object): """買票類及實現方法""" def __init__(self, user_name, password, passengers, from_time, from_station, to_station, number, seat_type, receiver_mobile, receiver_email): """定義例項屬性,初始化""" # 1206賬號密碼 self.user_name = user_name self.password = password # 乘客姓名 self.passengers = passengers # 起始站和終點站 self.from_station = from_station self.to_station = to_station # 乘車日期 self.from_time = from_time # 車次編號 self.number = number.capitalize() # 座位型別所在td位置 if seat_type == '商務座特等座': seat_type_index = 1 seat_type_value = 9 elif seat_type == '一等座': seat_type_index = 2 seat_type_value = 'M' elif seat_type == '二等座': seat_type_index = 3 seat_type_value = 0 elif seat_type == '高階軟臥': seat_type_index = 4 seat_type_value = 6 elif seat_type == '軟臥': seat_type_index = 5 seat_type_value = 4 elif seat_type == '動臥': seat_type_index = 6 seat_type_value = 'F' elif seat_type == '硬臥': seat_type_index = 7 seat_type_value = 3 elif seat_type == '軟座': seat_type_index = 8 seat_type_value = 2 elif seat_type == '硬座': seat_type_index = 9 seat_type_value = 1 elif seat_type == '無座': seat_type_index = 10 seat_type_value = 1 elif seat_type == '其他': seat_type_index = 11 seat_type_value = 1 else: seat_type_index = 7 seat_type_value = 3 self.seat_type_index = seat_type_index self.seat_type_value = seat_type_value # 通知資訊 self.receiver_mobile = receiver_mobile self.receiver_email = receiver_email # 主要頁面網址 self.login_url = 'https://kyfw.12306.cn/otn/login/init' self.init_my_url = 'https://kyfw.12306.cn/otn/index/initMy12306' self.ticket_url = 'https://kyfw.12306.cn/otn/leftTicket/init' # 瀏覽器驅動資訊,驅動下載頁:https://sites.google.com/a/chromium.org/chromedriver/downloads self.driver_name = 'chrome' self.executable_path = 'C:\\Users\cuizy\AppData\Local\Programs\Python\Python36\Scripts\chromedriver.exe' def do_login(self): """登入功能實現,手動識別驗證碼進行登入""" self.driver.visit(self.login_url) sleep(1) self.driver.fill('loginUserDTO.user_name', self.user_name) self.driver.fill('userDTO.password', self.password) print('請輸入驗證碼……') while True: if self.driver.url != self.init_my_url: sleep(1) else: break def start_brush(self): """買票功能實現""" self.driver = Browser(driver_name=self.driver_name, executable_path=self.executable_path) # 瀏覽器視窗的大小 self.driver.driver.set_window_size(900, 700) self.do_login() self.driver.visit(self.ticket_url) try: print('開始刷票……') # 載入車票查詢資訊 self.driver.cookies.add({"_jc_save_fromStation": self.from_station}) self.driver.cookies.add({"_jc_save_toStation": self.to_station}) self.driver.cookies.add({"_jc_save_fromDate": self.from_time}) self.driver.reload() count = 0 while self.driver.url.split('?')[0] == self.ticket_url: self.driver.find_by_text('查詢').click() sleep(1) count += 1 print('第%d次點選查詢……' % count) try: car_no_location = self.driver.find_by_id("queryLeftTable")[0].find_by_text(self.number)[1] current_tr = car_no_location.find_by_xpath("./../../../../..") if current_tr.find_by_tag('td')[self.seat_type_index].text == '--': print('無此座位型別出售,已結束當前刷票,請重新開啟!') sys.exit(1) elif current_tr.find_by_tag('td')[self.seat_type_index].text == '無': print('無票,繼續嘗試……') else: # 有票,嘗試預訂 print('刷到票了(餘票數:' + str(current_tr.find_by_tag('td')[self.seat_type_index].text) + '),開始嘗試預訂……') current_tr.find_by_css('td.no-br>a')[0].click() sleep(1) key_value = 1 for p in self.passengers: # 選擇使用者 print('開始選擇使用者……') self.driver.find_by_text(p).last.click() # 選擇座位型別 print('開始選擇席別……') if self.seat_type_value != 0: seat_select = self.driver.find_by_id("seatType_" + str(key_value))[0] seat_select.find_by_xpath("//option[@value='" + str(self.seat_type_value) + "']")[0].click() key_value += 1 sleep(0.5) if p[-1] == ')': self.driver.find_by_id('dialog_xsertcj_ok').click() print('正在提交訂單……') self.driver.find_by_id('submitOrder_id').click() sleep(2) # 檢視放回結果是否正常 submit_false_info = self.driver.find_by_id('orderResultInfo_id')[0].text if submit_false_info != '': print(submit_false_info) self.driver.find_by_id('qr_closeTranforDialog_id').click() sleep(0.2) self.driver.find_by_id('preStep_id').click() sleep(0.3) continue print('正在確認訂單……') self.driver.find_by_id('qr_submit_id').click() print('預訂成功,請及時前往支付……') # 傳送通知資訊 self.send_mail(self.receiver_email, '恭喜您,搶到票了,請及時前往12306支付訂單!') self.send_sms(self.receiver_mobile, '您的驗證碼是:8888。請不要把驗證碼洩露給其他人。') except Exception as error_info: print(error_info) except Exception as error_info: print(error_info) def send_sms(self, mobile, sms_info): """傳送手機通知簡訊,用的是-互億無線-的測試簡訊""" host = "106.ihuyi.com" sms_send_uri = "/webservice/sms.php?method=Submit" account = "C59782899" pass_word = "19d4d9c0796532c7328e8b82e2812655" params = parse.urlencode( {'account': account, 'password': pass_word, 'content': sms_info, 'mobile': mobile, 'format': 'json'} ) headers = {"Content-type": "application/x-www-form-urlencoded", "Accept": "text/plain"} conn = httplib2.HTTPConnectionWithTimeout(host, port=80, timeout=30) conn.request("POST", sms_send_uri, params, headers) response = conn.getresponse() response_str = response.read() conn.close() return response_str def send_mail(self, receiver_address, content): """傳送郵件通知""" # 連線郵箱伺服器資訊 host = 'smtp.163.com' port = 25 sender = '******@163.com' # 你的發件郵箱號碼 pwd = '******' # 不是登陸密碼,是客戶端授權密碼 # 發件資訊 receiver = receiver_address body = '<h2>溫馨提醒:</h2><p>' + content + '</p>' msg = MIMEText(body, 'html', _charset="utf-8") msg['subject'] = '搶票成功通知!' msg['from'] = sender msg['to'] = receiver s = smtplib.SMTP(host, port) # 開始登陸郵箱,併發送郵件 s.login(sender, pwd) s.sendmail(sender, receiver, msg.as_string()) if __name__ == '__main__': # 12306使用者名稱 user_name = input('請輸入12306使用者名稱:') while user_name == '': user_name = input('12306使用者名稱不能為空,請重新輸入:') # 12306登陸密碼 password = input('請輸入12306登陸密碼:') while password == '': password = input('12306登陸密碼不能為空,請重新輸入:') # 乘客姓名 passengers_input = input('請輸入乘車人姓名,多人用英文逗號“,”連線,(例如單人“張三”或者多人“張三,李四”):') passengers = passengers_input.split(",") while passengers_input == '' or len(passengers) > 4: print('乘車人最少1位,最多4位!') passengers_input = input('請重新輸入乘車人姓名,多人用英文逗號“,”連線,(例如單人“張三”或者多人“張三,李四”):') passengers = passengers_input.split(",") # 乘車日期 from_time = input('請輸入乘車日期(例如“2018-08-08”):') date_pattern = re.compile(r'^\d{4}-\d{2}-\d{2}$') while from_time == '' or re.findall(date_pattern, from_time) == []: from_time = input('乘車日期不能為空或者時間格式不正確,請重新輸入:') # 城市cookie字典 city_list = { 'bj': '%u5317%u4EAC%2CBJP', # 北京 'hd': '%u5929%u6D25%2CTJP', # 邯鄲 'nn': '%u5357%u5B81%2CNNZ', # 南寧 'wh': '%u6B66%u6C49%2CWHN', # 武漢 'cs': '%u957F%u6C99%2CCSQ', # 長沙 'ty': '%u592A%u539F%2CTYV', # 太原 'yc': '%u8FD0%u57CE%2CYNV', # 運城 'gzn': '%u5E7F%u5DDE%u5357%2CIZQ', # 廣州南 'wzn': '%u68A7%u5DDE%u5357%2CWBZ', # 梧州南 } # 出發站 from_input = input('請輸入出發站,只需要輸入首字母就行(例如北京“bj”):') while from_input not in city_list.keys(): from_input = input('出發站不能為空或不支援當前出發站(如有需要,請聯絡管理員!),請重新輸入:') from_station = city_list[from_input] # 終點站 to_input = input('請輸入終點站,只需要輸入首字母就行(例如北京“bj”):') while to_input not in city_list.keys(): to_input = input('終點站不能為空或不支援當前終點站(如有需要,請聯絡管理員!),請重新輸入:') to_station = city_list[to_input] # 車次編號 number = input('請輸入車次號(例如“G110”):') while number == '': number = input('車次號不能為空,請重新輸入:') # 座位型別 seat_type = input('請輸入座位型別(例如“軟臥”):') while seat_type == '': seat_type = input('座位型別不能為空,請重新輸入:') # 搶票成功,通知該手機號碼 receiver_mobile = input('請預留一個手機號碼,方便搶到票後進行通知(例如:18888888888):') mobile_pattern = re.compile(r'^1{1}\d{10}$') while receiver_mobile == '' or re.findall(mobile_pattern, receiver_mobile) == []: receiver_mobile = input('預留手機號碼不能為空或者格式不正確,請重新輸入:') receiver_email = input('請預留一個郵箱,方便搶到票後進行通知(例如:[email protected]):') while receiver_email == '': receiver_email = input('預留郵箱不能為空,請重新輸入:') # 開始搶票 ticket = BrushTicket(user_name, password, passengers, from_time, from_station, to_station, number, seat_type, receiver_mobile, receiver_email) ticket.start_brush()
最新版的原始碼和具體操作方法都在我的QQ裙上,大家有想要的可以直接來領取,推薦下我自己建的Python學習群:609616831,群裡都是學Python的,如果你想學或者正在學習Python ,歡迎互相交流學習,大家都是軟體開發黨,不定期分享乾貨(只有Python軟體開發相關的),包括我自己整理的一份2020最新的Python進階資料,如果對大家有幫助的話,記得一鍵三連哦!