手把手|100行Python代码自动抢火车票!(包教包会)

浏览: 2271

作者:菜鸟分析

个人介绍:一个痴迷于Python语言的业余程序猿,未来的理想是能够与一群痴迷于Python语言的程序猿改变世界

知乎专栏: https://zhuanlan.zhihu.com/c_149865214

专栏介绍:恋习Python|因痴恋Python而起,因学习Python而聚,与大家一起疯狂练习Python代码

image.png

又到一年一度春运大会,2017年春运抢火车票还是那么难,各大互联网公司都推出抢票服务,只要加钱给服务费就可以增加抢到票的几率,有些代售火车票点,说给100元服务费,可以帮抢到,看来这水很深啊!

下面我们利用自己学的技术来自动抢票,本次脚本基于Python3.6+splinter来实现:

首先介绍一下splinter使用:

splinter.brower是一个开源工具,通过Python自动化测试web,通过电脑自动操作网页。

Splinter模块是python egg,下载当然很简单,安装:pip install splinter

同时还需要浏览器的驱动,Splinter的Browser类默认优先调用的驱动是firefox,所以用chrome的话需要在初始化Browser时候指定driver_name="chrome"参数,建议都明确指定浏览器!

注意:Chrome的驱动chromedriver,注意版本要对应,不然基本上会有unknown error,打不开浏览器

splinter.brower基础知识:

  • 创建一个Browser实例,就会打开相应的浏览器。

  • visit(url): 故名思议,访问指定网站

  • find_by_id("控件的id").first: 根据控件的属性id找到控件,一般控件都有独立唯一的id。不然,Splinter api还提供by_name,by_id,by_tag等方法!first表示返回第一次找到的控件。

  • fill("要填充的内容"): 用指定的内容填充相应控件

    • 控件是指对数据和方法的封装。控件可以有自己的属性和方法,其中属性是控件数据的简单访问者,方法则是控件的一些简单而可见的功能、控件创建过程包括设计、开发、调试(就是所谓的3Ds开发流程,即Design、Develop、Debug)工作, 然后是控件的使用。

    • 设计控件是一项繁重的工作。自行开发控件与使用控件进行可视化程序开发存在着极大的不同,要求程序员精通面向对象程序设计。创建控件的最大意义在于封装重复的工作,其次是可以扩充现有控件的功能。

  • click(): 点击控件

  • 登录后,browser.cookies.all()中保存了本次登录的cookie信息(dict类型),可以打印出来或者保存下次使用

  • quit_browser(browser)函数: 要求用户交互输入q再退出。否则,程序跑完之后就直接退出了,释放Browser的实例,调用quit()方法,浏览器也就关闭了。

  • reload() 方法用于重新加载当前文档

实现思路:

首先我们需要登陆12306网站,登录时需要输入用户名密码,然后需要输入蛋疼的验证码,然后选择起、始站,时间,车次类型,点击查询,再选择车次,乘客,提交订单。如果按照这样的手动操作下来,票早已经没有了

备注:加粗字体都是需要购买火车票的属性

实现目标:

整个流程全自动,自动登陆,自动查询,自动订单,自动提交订单( (暂时不实现自动点击验证码,验证码成功几率比较低)

思维导图:

image.png

整体思路就是这样,基本可以实现自动抢票,唯一不足就是不能自动选取硬座、硬卧等车座类型;

总感觉不完美,有大神懂得可以指教一下,有没有好的思路,谢谢!

对,朋友看了还不会的或者运营过程中有出现错误的,可以私信我,我承诺的包教包会哦(哈哈哈)

附详情代码如下:

from splinter.browser import Browser
from time import sleep
import traceback


class Buy_Tickets(object):
   # 定义实例属性,初始化
   def __init__(self, username, passwd, order, passengers, dtime, starts, ends):
       self.username = username
       self.passwd = passwd
       # 车次,0代表所有车次,依次从上到下,1代表所有车次,依次类推
       self.order = order
       # 乘客名
       self.passengers = passengers
       # 起始地和终点
       self.starts = starts
       self.ends = ends
       # 日期
       self.dtime = dtime
       # self.xb = xb
       # self.pz = pz
       self.login_url = 'https://kyfw.12306.cn/otn/login/init'
       self.initMy_url = 'https://kyfw.12306.cn/otn/index/initMy12306'
       self.ticket_url = 'https://kyfw.12306.cn/otn/leftTicket/init'
       self.driver_name = 'chrome'
       self.executable_path = 'C:\Python36\Scripts\chromedriver.exe'
   # 登录功能实现
   def login(self):
       self.driver.visit(self.login_url)
       self.driver.fill('loginUserDTO.user_name', self.username)
       # sleep(1)
       self.driver.fill('userDTO.password', self.passwd)
       # sleep(1)
       print('请输入验证码...')
       while True:
           if self.driver.url != self.initMy_url:
               sleep(1)
           else:
               break
   # 买票功能实现
   def start_buy(self):
       self.driver = Browser(driver_name=self.driver_name, executable_path=self.executable_path)
       #窗口大小的操作
       self.driver.driver.set_window_size(700, 500)
       self.login()
       self.driver.visit(self.ticket_url)
       try:
           print('开始购票...')
           # 加载查询信息
           self.driver.cookies.add({"_jc_save_fromStation": self.starts})
           self.driver.cookies.add({"_jc_save_toStation": self.ends})
           self.driver.cookies.add({"_jc_save_fromDate": self.dtime})
           self.driver.reload()
           count = 0
           if self.order != 0:
               while self.driver.url == self.ticket_url:
                   self.driver.find_by_text('查询').click()
                   count += 1
                   print('第%d次点击查询...' % count)
                   try:
                       self.driver.find_by_text('预订')[self.order-1].click()
                       sleep(1.5)
                   except Exception as e:
                       print(e)
                       print('预订失败...')
                       continue
           else:
               while self.driver.url == self.ticket_url:
                   self.driver.find_by_text('查询').click()
                   count += 1
                   print('第%d次点击查询...' % count)
                   try:
                       for i in self.driver.find_by_text('预订'):
                           i.click()
                           sleep(1)
                   except Exception as e:
                       print(e)
                       print('预订失败...')
                       continue
           print('开始预订...')
           sleep(1)
           print('开始选择用户...')
           for p in self.passengers:

               self.driver.find_by_text(p).last.click()
               sleep(0.5)
               if p[-1] == ')':
                   self.driver.find_by_id('dialog_xsertcj_ok').click()
           print('提交订单...')
           # sleep(1)
           # self.driver.find_by_text(self.pz).click()
           # sleep(1)
           # self.driver.find_by_text(self.xb).click()
           # sleep(1)
           self.driver.find_by_id('submitOrder_id').click()
           sleep(2)
           print('确认选座...')
           self.driver.find_by_id('qr_submit_id').click()
           print('预订成功...')
       except Exception as e:
           print(e)




if __name__ == '__main__':
   # 用户名
   username = 'xxxx'
   # 密码
   password = 'xxx'
   # 车次选择,0代表所有车次
   order = 2
   # 乘客名,比如passengers = ['丁小红', '丁小明']
   # 学生票需注明,注明方式为:passengers = ['丁小红(学生)', '丁小明']
   passengers = ['丁彦军']
   # 日期,格式为:'2018-01-20'
   dtime = '2018-01-19'
   # 出发地(需填写cookie值)
   starts = '%u5434%u5821%2CWUY' #吴堡
   # 目的地(需填写cookie值)
   ends = '%u897F%u5B89%2CXAY' #西安

   # xb =['硬座座']
   # pz=['成人票']


   Buy_Tickets(username, password, order, passengers, dtime, starts, ends).start_buy()

Python爱好者社区历史文章大合集

Python爱好者社区历史文章列表(每周append更新一次)

福利:文末扫码立刻关注公众号,“Python爱好者社区”,开始学习Python课程:

关注后在公众号内回复课程即可获取:

小编的Python入门视频课程!!!

崔老师爬虫实战案例免费学习视频。

丘老师数据科学入门指导免费学习视频。

陈老师数据分析报告制作免费学习视频。

玩转大数据分析!Spark2.X+Python 精华实战课程免费学习视频。

丘老师Python网络爬虫实战免费学习视频。

image.png

推荐 6
本文由 Python爱好者社区 创作,采用 知识共享署名-相同方式共享 3.0 中国大陆许可协议 进行许可。
转载、引用前需联系作者,并署名作者且注明文章出处。
本站文章版权归原作者及原出处所有 。内容为作者个人观点, 并不代表本站赞同其观点和对其真实性负责。本站是一个个人学习交流的平台,并不用于任何商业目的,如果有任何问题,请及时联系我们,我们将根据著作权人的要求,立即更正或者删除有关内容。本站拥有对此声明的最终解释权。

0 个评论

要回复文章请先登录注册