春节到了 教你使用python来抢票回家


Posted in Python onJanuary 06, 2020

这篇文章主要介绍了春节到了 教你使用python来抢票回家,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

不知不觉,一年一度的春运抢票大幕已经拉开,想快速抢到回家的车票吗?作为程序员,这些技术手段,你一定要知道。

为了让大家更快捷更便利的抢火车票,各种各样的抢票软件应需而生,这类软件大部分都是付费抢票的机制。

作为程序员,如何用技术手段抢到回家的票?来看看用 Python 写的抢票脚本。

手把手教你用 Python 抢票回家过年

环境介绍

  • windows 8.1
  • python3.6.1
  • firefox插件 geckodriver.exe

操作步骤

引入要的模块

from selenium import webdriver   #控制浏览器
    from selenium.webdriver.common.keys import Keys #用于给元素赋值
    import time  #时间模块
    from selenium.webdriver.support.select import Select #控制下拉框模块
    from selenium.webdriver.common.by import By  #寻找元素模块
    from selenium.webdriver.support.ui import WebDriverWait #“显示等待”模块
    from selenium.webdriver.support import expected_conditions as EC #等待条件模块

登陆模块

首先需要选择使用的浏览器,此处以 firefox 为例,下载:geckodriver.exe 。

提到的 stations.txt 可以直接看这个:

车站信息:

https://kyfw.12306.cn/otn/resources/js/framework/station_name.js?station_version=1.9042

将 geckodriver.exe 放到 python.exe 同级目录下即可(如果有报错的情况下,放一个该文件到与 firefox.exe 同级目录下,并添加环境变量)

#可以用input,也可以直接放入到后面的用户名、密码输入框中
    #可以利用标准输入进行批量的操作,此处以个人抢票操作为例
    # username = str(input('请输入你的用户名:'))
    # password = str(input('请输入你的密码:')) #这两行可以暂时忽略

    browser = webdriver.Firefox()   #驱动firefox浏览器
    browser.get("https://kyfw.12306.cn/otn/login/init")  #启动浏览器后进入该链接下
    browser.find_element_by_id('username').clear()
    browser.find_element_by_id('username').send_keys(‘xxxxx')  #xxxxx更换为用户名 
    browser.find_element_by_id('password').send_keys(‘xxxxx')  #xxxxx更换为密码  
    time.sleep(10)  #此时验证码自行点击,该处设置10秒延迟,可以自己设置
    try:
        browser.find_element_by_id('loginSub').click()   #点击登陆操作,该id为登陆按钮
        #或者 browser.find_element_by_link_text('登陆').click() #标签显示的名称
    except:
        browser.find_element_by_class_name('touclick-bgimg touclick-reload touclick-reload-normal').click()   #try中验证码输入点错了会在此处刷新一次
        time.sleep(20)           #第二次输入验证码前等待20秒,可以自己设置,第一次输入无误直接跳过
        browser.find_element_by_id('loginSub').click()  #重新输入验证码后的点击登陆

跳转模块

#默认跳转到首页
    time.sleep(2)  #此处一般无需设置时间等待,调试代码时使用
    clickReserve = browser.find_element_by_link_text('车票预订').click() #跳转到车票预定页面,该页面可以查询票
    time.sleep(2)    #出发地点和到达地点设置
    #此处value值为出发时刻的地点,BJP表示北京,更改value值在页面上不加载,基本不耗时间,从页面中也看不到出发地和目的地
    #此处内容以爬取,保存在stations.txt中,每行表示一个地址,打开文档ctrl + F查找即可
    jsf = 'var a = document.getElementById("fromStation");a.value = "BJP"'  #此处将BJP更换为你需要的出发地址,value值在以爬取到stations.txt中,自行查看
    browser.execute_script(jsf)
    jst = 'var a = document.getElementById("toStation");a.value = "LZJ"'  #终点,同上方法
    browser.execute_script(jst)
    js = "document.getElementById('train_date').removeAttribute('readonly')"  #时间选择时默认为只读,通过JS移除只读属性
    browser.execute_script(js)  #执行JS语句
    browser.find_element_by_id('train_date').clear()  #时间元素中默认有提示字,需要先清空
    browser.find_element_by_id('train_date').send_keys('2018-02-01')  #按照改格式输入需要查询的时间
    search = browser.find_element_by_id('query_ticket').click()  #输入好信息时点击查询,该处存在成人票和学生票,默认是成人票,如果购买,对学生票处执行以下语句即可:
    #browser.find_element_by_id('xxxx').click()  #对于id还是class或其它自行选择

开始购票

此处,就是点击预定的操作,我在这里只是举一个方法例子,也可以通过不断点击直到成功(这样可以避免网站倒计时和实际时间的时间差影响,但是不知道 12306 在抢票时对不断快速访问有没有限制)。

start_time = "Thu Jan 04 08:00:00 2018"  #首先设置需要抢票的时间
    b = time.mktime(time.strptime(start_time,"%a %b %d %H:%M:%S %Y"))    print(time.strftime("%a %b %d %H:%M:%S %Y", time.localtime(b)) ) #此处是为了调试代码使用,可忽略,不影响使用
    a = float(b)-time.time()  #利用自己设置的时间减去当前时间的时间戳
    time.sleep(a)  #上一步骤得出的秒数就是需要等待抢票的时间
try:   #此处本来有try中的部分就够了,WebDriverWait已有相应等待重复访问机制,默认为0.5秒试验一次,except中添加是为了以防万一
  WebDriverWait(browser,10).until(EC.presence_of_element_located((By.ID, "ticket_2400000Z550L")))  #查找需要预定的车次的id,直到出现,10表示共等待10秒
  ticket = browser.find_element_by_xpath('//tr[@id="ticket_2400000Z550L"]/td[13]/a').click()  #点击预定按钮except:
  browser.find_element_by_id('query_ticket').click()
  WebDriverWait(browser, 10).until(EC.presence_of_element_located((By.ID, "ticket_2400000Z550L")))
  ticket = browser.find_element_by_xpath('//tr[@id="ticket_2400000Z550L"]/td[13]/a').click()
"""
normalPassenger_8 数字表示该账号下的第几位,默认从0开始如果是第一个则为normalPassenger_0
"""WebDriverWait(browser,10).until(EC.presence_of_element_located((By.ID, "normalPassenger_8")))
browser.find_element_by_id('normalPassenger_8').click()  #id中的8表示账号下第九位s = Select(browser.find_element_by_id('seatType_1'))
s.select_by_value('6')  #此处value值看下方各个种类,6表示高级软卧browser.find_element_by_id('submitOrder_id').click()
WebDriverWait(browser,10).until(EC.presence_of_element_located((By.ID, "qr_submit_id")))
browser.find_element_by_link_text('提交订单')
browser.find_element_by_id('qr_submit_id').click()#-------------------------------------------------结束#硬座 1#硬卧 3#软卧 4#高级软卧 6#二等座 O(大写字母)#一等座 M#商务座 9

总结

需要替换的地方:

用户名,密码。

起始地点和目的地的 value 值,查 stations.txt 修改即可。

出发时间。

自己选择车次的 xpath 路径,路径不用变,变对应 id 即可。

勾选用户的位置(如果只要一个用户,默认用:normalPassenger_0)。

所选座位类别,默认为有票的类别里最便宜的种类。

其余的在测试中都相同,没有发现有变化,在使用前,可以测试一下代码,测试是注意注释掉提交订单的代码(下单有取消限制,每天好像只能取消三次),测试时网速正常。

有人说用浏览器执行速度会慢,确实对于可以直接识别验证码的脚本而言,没有界面的会更快一些,但是实际上所用时间为预定开始到结束,相同网络下,代码执行时间是要快于人工操作的,

另外,时间可以研究一下,之前研究过某宝的时间,秒杀时间是要比北京时间提前一点几秒的,感觉全国各地有微小时间差的。

完整脚本示例

#python3.6.1#data:2018-01-03#author:LGC247CG"""
说明:
1.该脚本主要是提供一个实现思路,实现方法有很多,可以优化的地方也有很多,触发机制也可以自己设置,代码以压缩到最短,只是为了让大家都可以看明白
2.正常网络状况下,不设置指定时间时,从点击确认验证码到下单基本上1秒左右,所以速度上还是没问题的
3.由于同时勾选多人和单人使用所需时间基本相同,希望该方法只用于技术交流,请勿作为黄牛使用
4.在作为技术交流的情况下,如果验证码可以实现将可以完全实现自动抢票:
--1>验证码有一定规律和数量,可以利用脚本获取所有图片,并加上相应标签
--2>将页面的文字和标签相匹配,再将图片进行相似度计算,对对应图片进行点击操作
--3>或是训练深度学习的图片识别模型,通过算法识别
"""from selenium import webdriverfrom selenium.webdriver.common.keys import Keysimport timefrom selenium.webdriver.support.select import Selectfrom selenium.webdriver.common.by import Byfrom selenium.webdriver.support.ui import WebDriverWaitfrom selenium.webdriver.support import expected_conditions as EC

browser = webdriver.Firefox()
browser.get("https://kyfw.12306.cn/otn/login/init")
browser.find_element_by_id('username').clear()
browser.find_element_by_id('username').send_keys('xxxxxxx')
browser.find_element_by_id('password').send_keys('xxxxxxx')
time.sleep(10)try:
  browser.find_element_by_id('loginSub').click()except:
  browser.find_element_by_class_name('touclick-bgimg touclick-reload touclick-reload-normal').click()
  time.sleep(15)
  browser.find_element_by_id('loginSub').click()#跳转到车票预定页面time.sleep(2)
clickReserve = browser.find_element_by_link_text('车票预订').click()#出发地点和到达地点设置WebDriverWait(browser,10).until(EC.presence_of_element_located((By.ID, "fromStation")))
jsf = 'var a = document.getElementById("fromStation");a.value = "BJP"'browser.execute_script(jsf)
jst = 'var a = document.getElementById("toStation");a.value = "LZJ"'browser.execute_script(jst)
js = "document.getElementById('train_date').removeAttribute('readonly')"browser.execute_script(js)
browser.find_element_by_id('train_date').clear()
browser.find_element_by_id('train_date').send_keys('2018-02-02')
search = browser.find_element_by_id('query_ticket').click()#对于时间,我一直觉得网站计算的时间和自己获取的时间差一秒左右,这个根据不同环境自己测试start_time = "Thu Jan 04 10:00:00 2018"  #首先设置需要抢票的时间b = time.mktime(time.strptime(start_time,"%a %b %d %H:%M:%S %Y"))
print(time.strftime("%a %b %d %H:%M:%S %Y", time.localtime(b)) ) #此处是为了调试代码使用,可忽略,不影响使用a = float(b)-time.time()  #利用自己设置的时间减去当前时间的时间戳time.sleep(a)  #上一步骤得出的秒数就是需要等待抢票的时间browser.find_element_by_id('query_ticket').click()  #时间到了先点击查询刷新一下,以防找不到元素try:
  WebDriverWait(browser,10).until(EC.presence_of_element_located((By.ID, "ticket_2400000Z550L")))
  ticket = browser.find_element_by_xpath('//tr[@id="ticket_2400000Z550L"]/td[13]/a').click()except:
  browser.find_element_by_id('query_ticket').click()
  WebDriverWait(browser, 10).until(EC.presence_of_element_located((By.ID, "ticket_250000K8880L")))
  ticket = browser.find_element_by_xpath('//tr[@id="ticket_250000K8880L"]/td[13]/a').click()"""
normalPassenger_8 数字表示该账号下的第几位,默认从0开始如果是第一个则为normalPassenger_0
"""WebDriverWait(browser,10).until(EC.presence_of_element_located((By.ID, "normalPassenger_8")))
browser.find_element_by_id('normalPassenger_8').click()
s = Select(browser.find_element_by_id('seatType_1'))
s.select_by_value('6')
browser.find_element_by_id('submitOrder_id').click()
WebDriverWait(browser,10).until(EC.presence_of_element_located((By.ID, "qr_submit_id")))
browser.find_element_by_link_text('提交订单')#browser.find_element_by_id('qr_submit_id').click()

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python读写Redis数据库操作示例
Mar 18 Python
python实现从网络下载文件并获得文件大小及类型的方法
Apr 28 Python
Python找出文件中使用率最高的汉字实例详解
Jun 03 Python
如何用itertools解决无序排列组合的问题
May 18 Python
python 实现调用子文件下的模块方法
Dec 07 Python
python用pandas数据加载、存储与文件格式的实例
Dec 07 Python
Flask框架实现的前端RSA加密与后端Python解密功能详解
Aug 13 Python
python requests包的request()函数中的参数-params和data的区别介绍
May 05 Python
python 删除excel表格重复行,数据预处理操作
Jul 06 Python
Django rest framework分页接口实现原理解析
Aug 21 Python
记一次python 爬虫爬取深圳租房信息的过程及遇到的问题
Nov 24 Python
Python获取字典中某个key的value
Apr 13 Python
Tensorflow 自定义loss的情况下初始化部分变量方式
Jan 06 #Python
在flask中使用python-dotenv+flask-cli自定义命令(推荐)
Jan 05 #Python
通过 Python 和 OpenCV 实现目标数量监控
Jan 05 #Python
python中count函数简单用法
Jan 05 #Python
Python猴子补丁知识点总结
Jan 05 #Python
Python调用Windows API函数编写录音机和音乐播放器功能
Jan 05 #Python
django ajax发送post请求的两种方法
Jan 05 #Python
You might like
php数据库抽象层 PDO
2011/05/07 PHP
PHP中使用array函数新建一个数组
2015/11/19 PHP
thinkphp 字母函数详解T/I/N/D/M/A/R/U
2017/04/03 PHP
javascript判断iphone/android手机横竖屏模式的函数
2011/12/20 Javascript
jquery 延迟执行实例介绍
2013/08/20 Javascript
JS+CSS实现自动切换的网页滑动门菜单效果代码
2015/09/14 Javascript
通用javascript代码判断版本号是否在版本范围之间
2015/11/29 Javascript
APP中javascript+css3实现下拉刷新效果
2016/01/27 Javascript
domReady的实现案例
2016/11/23 Javascript
JavaScript中双向数据绑定详解
2017/05/03 Javascript
JavaScript设置名字输入不合法的实现方法
2017/05/23 Javascript
Angular 2.0+ 的数据绑定的实现示例
2017/08/09 Javascript
JS+CSS实现滚动数字时钟效果
2017/12/25 Javascript
vue数据传递--我有特殊的实现技巧
2018/03/20 Javascript
使用javascript函数编写简单银行取钱存钱流程
2018/05/26 Javascript
解决iview多表头动态更改列元素发生的错误的方法
2018/11/02 Javascript
详解angular2 控制视图的封装模式
2018/12/27 Javascript
详解jquery和vue对比
2019/04/16 jQuery
微信小程序中如何使用flyio封装网络请求
2019/07/03 Javascript
浅析Vue中拆分视图层代码的5点建议
2019/08/15 Javascript
JavaScript This指向问题详解
2019/11/25 Javascript
[01:16:16]DOTA2-DPC中国联赛定级赛 RNG vs Phoenix BO3第二场 1月8日
2021/03/11 DOTA
Python使用pymysql小技巧
2017/06/04 Python
Python字符串格式化%s%d%f详解
2018/02/02 Python
Python爬虫——爬取豆瓣电影Top250代码实例
2019/04/17 Python
python爬虫 urllib模块反爬虫机制UA详解
2019/08/20 Python
Python opencv相机标定实现原理及步骤详解
2020/04/09 Python
纯DOM+CSS3实现简单的小风车动画
2016/09/27 HTML / CSS
幼儿园教师个人总结
2015/02/05 职场文书
2015年乡镇扶贫工作总结
2015/04/08 职场文书
试用期旷工辞退通知书
2015/04/17 职场文书
新闻稿件写作范文
2015/07/18 职场文书
新入职员工工作总结
2015/10/15 职场文书
预备党员的思想汇报,你真的会写吗?
2019/06/28 职场文书
大学生暑期社会实践的个人总结!
2019/07/17 职场文书
jquery插件实现搜索历史
2021/04/24 jQuery