前言

上篇写了12306登录,隔了快一个月了,才准备动手写下单篇,真的要非常感谢博客园的 Asimple朋友,如果不是看到你的留言,我几乎都忘了要写下篇了,这一点在简书上就不好,都没人看/(ㄒoㄒ)/~~,刚开始写博客,真的需要大家的鼓励,看的人多了自然有动力写更多的,所以这一篇要给那些看过我上篇的同学们,尤其是这位Asimple同学,就是为你而写,没错就是这个原因。因为你让我知道了有人在看,而且用心的在看。发了这么多感慨,其实我不是这么爱感慨的人(✿◡‿◡),直入主题吧。

还得说一下这次我用的是Firefox浏览器,在上篇中说了Chrome浏览器的一个问题,在抓取的请求过多的情况下,前面的请求就可能看不到请求信息,最后搞得我不得不搞了个虚拟机,装了xp,用Fiddler去查看请求信息,后来就换了Firefox,最新版的Firefox没有Firebug可用,但是自带的开发者工具足够用了。当然Chrome我也没卸载,因为Chrome上有个很赞的功能,在元素界面查找元素的时候可以用css和xpath,可不是右键copy里面的css和xpath哦 
 
在这个查找框里你就可以写自己的css和xpath,看实时效果,这个真是太棒了,不用装额外的插件。Firefox上暂时还没有发现这个功能,也不知道有没有类似的插件,如果有人知道的话,麻烦回复一下,先谢过了。鼠标左键还坏了,只能把右键设置一下暂时用着,一波三折呀!这次真的要进入主题了。 

卧了一个槽,忽略我上面说的吧,要写博客了,有一个请求Firefox竟然不给显示了,还是虚拟机吧


后记

后记为什么要加载前言后面,而不是在文章最后,我怕你不看。在这次分析中我基本没有添加什么代码,因为基本上每一个请求就是定义一个字典、一个url,然后发送请求,获取数据,然后继续下一个。另一方面就是我并没有做代码优化和整理。我们都应该知道对于没有反爬措施的网站,基本上看两个小时的爬虫教程就能写的出来,对于有反爬的网站,最难得地方是分析阶段,而不是发送请求。最后一点就是做这个也是一时兴起,积累一下经验,最主要是开始写一写博客。同时给像我一样初学爬虫的朋友一个例子、一个思路。


查票


车票预订界面的url:https://kyfw.12306.cn/otn/leftTicket/init,选好票以后点击查询 

多了一下两个请求,第一个请求我没用,没有任何影响,我们就不用去管它,直接看第二个请求 
https://kyfw.12306.cn/otn/leftTicket/query?leftTicketDTO.train_date=2017-12-25&leftTicketDTO.from_station=BJP&leftTicketDTO.to_station=SHH&purpose_codes=ADULT 

看一下参数和返回数据,太乱了,稍微仔细看一下,预订、有、G5,好像还是有些有用信息的,可以按|拆分一下看看 

看到了车次G101,8、9是开车时间和到达时间,10是历时,商务座特等座9张余票,32行显示9,一等二等座都是有票,对应30、31,就不能具体确定了,暂时先不管,至少我们确定了这个请求是查询出我们需要的车票信息了,那么再看一下请求参数,第一个是时间很容易理解,第四个好像是票的类型,成人票,翻译一下单词就知道了,反正每次都一样,不用管了,中间两个出发站、目的地,不过这些字母是啥意思,应该是站名对应的编码,在这个请求之前肯定是有对应关系的, 

https://kyfw.12306.cn/otn/resources/js/framework/station_name.js?station_version=1.9034 
一个js的请求@bjb|北京北|VAP|beijingbei|bjb|0,拆分以后可以看到站名和编号

def query_ticket_info(date, info):
    js_info = json.loads(info)
    if js_info.get("status") != True:
        print("查询余票失败")
        return

result = js_info.get("data").get("result")
    for i in result:
        lst = i.split('|')
        if lst[11] != "Y": #主要是判断是否开票了,见下图情况
            continue
        item = {
            "预定号":lst[0], #预定号
            "train_no":lst[2],
            "车次":lst[3], #车次
            "始发站":get_station_name_from_code(lst[4]), #始发站
            "终点站":get_station_name_from_code(lst[5]), #终点站
            "起始站":get_station_name_from_code(lst[6]), #起始站
            "目标站":get_station_name_from_code(lst[7]), #目标站
            "出发时间":"{} {}".format(date, lst[8]), #出发时间
            "到达时间":get_end_time(date, lst[8], lst[9], lst[10]), #到达时间
            "历时":lst[10], #历时
            "train_location":lst[15],
            "高级动卧":lst[21], #高级动卧
            "软卧":lst[23], #软卧
            "软座":lst[24], #软座
            "特等座":lst[25], #特等座
            "无座":lst[26], #无座
            "硬卧":lst[28], #硬卧
            "硬座":lst[29], #硬座
            "二等座":lst[30], #二等座
            "一等座":lst[31], #一等座
            "商务座":lst[32], #商务座
            "动卧":lst[33], #动卧
        }
        yield item

这个函数需要解释一下,主要是这些站点信息是怎么找到的,说一下思路 
- 当我们打开车票预订界面的时候,是这样的,下面是空的 

- 点击查询后 

- 由此我们知道车票信息是通过js动态添加的(maybe),那么我们就查找动态创建出来的元素,随便在上面找一个元素特征,比如我找这个 
 
当然你可以随便找一个,前提是不是动态创建的,然后在我们的js和document请求中去查找这个值,最后我找到了https://kyfw.12306.cn/otn/resources/merged/queryLeftTicket_end_UAM_js.js?scriptVersion=1.9053 
 
我找这个是干什么呢?我们上面的请求得到了余票信息,是json格式的,那么肯定是通过js把它添加到界面上,我们要知道哪一个信息是添加到硬座上,哪一个是添加到硬卧上,就像我代码里写的那样,把所有的票种都找出来 
- 通过分析js 
 
然后把大部分我们需要的参数都对应出来,这样就看到像我上面写的函数那样,取出我们需要的信息 
- 其实有一种更简单的方式,就是你查询了余票信息以后,会看到余票几张几张,然后去对应的信息参数中去找,比如 
 
我们就知道了32对应的是商务座特等,多查询一些站点就会把所有对应索引都找出来


预订

点击预订后看一下请求,记住我上一遍说过的,一般是看xhr和document请求, 
https://kyfw.12306.cn/otn/login/checkUser 
看起来很简单,参数也只有一个_json_att,值为空 
重头戏来了https://kyfw.12306.cn/otn/leftTicket/submitOrderRequest 
看请求参数secretStr,其他的请求几次发现没啥变化,重点就在这个secretStr上了,太乱了,咦,我上面好像说过这三个字,对他们有关系。怎么去理解呢,这里是发了一个post请求,而这个secretStr是作为参数发送给服务器的,那么它必定是在我们本地产生的,在这个请求之前应该能找到。看一下特征:+wOQuwrBzvR6e…。是不是能发现查票那个请求里返回的数据第一个%2BwOQuwrBzvR6e,相似度很高啊,肯定是进行了编码或解码。Fiddler这点很好 
 
点击查票请求的第一条数据,右键->Send to TextWizare… 
 
哇哦,So Beautiful,这下就相等了,是使用了urldecode,这里支持很多种编码解码方式,非常方便,真相已经出来了,secretStr是我们上面输出的第0行字符串的urldecode解码值,python3中是parse.unquote。 
train_date:订哪一天的票,back_train_date:今天的时间,还有出发站和目的地。

12306自动刷票下单-查票下单(二)相关推荐

  1. 12306自动刷票下单-查票下单

    12306自动刷票下单-登录 前言 上篇写了12306登录,隔了快一个月了,才准备动手写下单篇,真的要非常感谢博客园的 Asimple朋友,如果不是看到你的留言,我几乎都忘了要写下篇了,这一点在简书上 ...

  2. 第二篇 12306自动刷票下单-查票下单

    前言 上篇写了12306登录,隔了快一个月了,才准备动手写下单篇,真的要非常感谢博客园的 Asimple朋友,如果不是看到你的留言,我几乎都忘了要写下篇了,这一点在简书上就不好,都没人看/(ㄒoㄒ)/ ...

  3. 12306自动刷票下单-登录篇

    12306网站推出图片验证码以后,对于抢票软件就提出了更高的要求,本篇并不涉及自动识别验证码登录(主要是博主能力所限),提供一个途径-打码平台,这个几乎是可以破解所有验证码了,本篇主要是分享一下123 ...

  4. 12306自动刷票下单-下单(三)

    12306自动刷票下单-登录篇(一) 12306自动刷票下单-查票预定(二) 下单 进入下单界面了  https://kyfw.12306.cn/otn/confirmPassenger/initDc ...

  5. 12306自动刷票下单-下单

    12306自动刷票下单-登录 12306自动刷票下单-查票预定 下单 进入下单界面了 https://kyfw.12306.cn/otn/confirmPassenger/initDc 还有一个请求h ...

  6. 第三篇 12306自动刷票下单-下单

    下单 进入下单界面了  https://kyfw.12306.cn/otn/confirmPassenger/initDc    还有一个请求https://kyfw.12306.cn/otn/con ...

  7. 12306自动刷票下单-登录篇(一)

    12306网站推出图片验证码以后,对于抢票软件就提出了更高的要求,本篇并不涉及自动识别验证码登录(主要是博主能力所限),提供一个途径-打码平台,这个几乎是可以破解所有验证码了,本篇主要是分享一下123 ...

  8. 基于Python 3.4 实现的12306查票器

    学了一学期Python还是让我感受到这是一门强大的语言,如果能熟练使用许多库的话,它给人带来的方便是非常大的, 很难想象一位Freelancer当初单枪匹马搞出来的语言,如今能发展壮大到这个程度! 本 ...

  9. python12306下单步骤_python+splinter实现12306网站刷票并自动购票流程

    通过python+splinter,实现在12306网站刷票并自动购票流程(无法自动识别验证码). 此类程序只是提高了12306网站的 刷新频率(默认自动查询的刷新频率为5秒).对于学习splinte ...

最新文章

  1. OpenFlow Switch — 1.3 规范
  2. Linux中使用ps、awk、sh一起批量杀死所有的dotnet进程。
  3. 图解NHibernate项目框架
  4. c语言设计四路彩灯显示系统,四路彩灯控制器设计方案.doc
  5. offset;scroll;client
  6. [PL/SQL]使用存储过程实现导出指定数据到文件(仿EXP)|转|
  7. pyTorch自然语言处理简单例子
  8. Android学习笔记---13_文件的操作模式.各个应用之间的文件权限
  9. Carryon的字符串
  10. 再学 GDI+[22]: TGPLinearGradientBrush - 之一: TLinearGradientMode
  11. UIProgressView的详细使用
  12. XCL-Charts图表库中柱形图的同源风格切换介绍
  13. 赶紧收藏!不可多得的Instagram运营技巧
  14. 电视和计算机共享视频,电脑中的图片视频一键共享到电视上去看
  15. java jconsole_关于java:JConsole在Linux中的位置
  16. 墨魂服务器维修,墨魂琅轩路线怎么选最新游戏攻略
  17. 教育知识与能力(中学)
  18. Python Network(三)案例(无向图,有向图,权重,点线分类与大小粗细)
  19. 工业相机之镜头基础知识
  20. APP开发流程实例讲解-儒释道网络电台八天开发全程-百度云深度兼容测试并进一步优化排错

热门文章

  1. java Condition类的详细介绍
  2. 我是游戏王 哎哟~这个网站不错哟!
  3. JAVA之读取二进制文件
  4. 提高PPT操作APM的小技巧
  5. 怎样构建笔记本虚拟wifi热点
  6. 咸阳职业技术学院计算机专业咋样,咸阳职业技术学院王牌专业有哪些、专业排名情况怎样...
  7. 4273. 【NOIP2015模拟10.28B组】圣章-精灵使的魔法语
  8. 日文xp系统中 日文键盘模式转英式键盘模式
  9. matlab画间断坐标轴,Matlab制图中如何实现坐标轴的打断
  10. oracle pls 00905,【案例】Oracle报错PLS-00714 PLS-00951原因和解决办法笔记