教你用python实现12306余票查询

作者:weixin_BeefpasteC 时间:2021-03-16 01:39:19 

python实现12306余票查询

我们说先在浏览器中打开开发者工具(F12),尝试一次余票的查询,通过开发者工具查看发出请求的包

教你用python实现12306余票查询

余票查询界面

可以看到红框框中的URL就是我们向12306服务器发出的请求,那么具体是什么呢?我们来看看

[
https://kyfw.12306.cn/otn/leftTicket/queryZ?leftTicketDTO.train_date=2019-01-21&leftTicketDTO.from_station=CDW&leftTicketDTO.to_station=SZQ&purpose_codes=ADULT](https://kyfw.12306.cn/otn/leftTicket/queryZ?leftTicketDTO.train_date=2019-01-21&leftTicketDTO.from_station=CDW&leftTicketDTO.to_station=SZQ&purpose_codes=ADULT)

可以看到发出请求的几个字段:

leftTicketDTO.train_date:查询的日期
leftTicketDTO.from_station:查询的出发地
leftTicketDTO.to_station:查询的目的地
purpose_codes:不太清楚这个字段是用来做什么的,就默认吧

可以从我们递交的URL请求看出,我们输入的成都,深圳都变成了对应的编号,比如,成都(CDW)、深圳(SZQ),所以当我们程序进行输入的时候要进行一下处理,12306的一个地方存储着这些城市名与编码对应的文档:

[
https://kyfw.12306.cn/otn/resources/js/framework/station_name.js?station_version=1.8971](https://kyfw.12306.cn/otn/resources/js/framework/station_name.js?station_version=1.8971)

教你用python实现12306余票查询

站点编码对应

下面我们就编写一个小程序,将这些城市名与编号提取出来:


   import re,requests
   url = "https://kyfw.12306.cn/otn/resources/js/framework/station_name.js?station_version=1.8971"
   response = requests.get(url,verify=False)
   #将车站的名字和编码进行提取
   chezhan = re.findall(r'([\u4e00-\u9fa5]+)\|([A-Z]+)', response.text)
   chezhan_code = dict(chezhan)
   #进行交换
   chezhan_names = dict(zip(chezhan_code.values(),chezhan_code.keys()))
   #打印出得到的车站字典
   print(chezhan_names)

得到的打印结果如下(只截取部分显示):

{‘VAP': ‘北京北', ‘BOP': ‘北京东', ‘BJP': ‘北京', ‘VNP': ‘北京南', ‘BXP': ‘北京西', ‘IZQ':
‘广州南', ‘CUW': ‘重庆北', ‘CQW': ‘重庆', ‘CRW': ‘重庆南', ‘CXW': ‘重庆西', ‘GGQ': ‘广州东',
‘SHH': ‘上海', ‘SNH': ‘上海南', ‘AOH': ‘上海虹桥', ‘SXH': ‘上海西', ‘TBP': ‘天津北', ‘TJP':
‘天津', ‘TIP': ‘天津南', ‘TXP': ‘天津西', ‘XJA': ‘香港西九龙', ‘CCT': ‘长春', ‘CET': ‘长春南',
‘CRT': ‘长春西', ‘ICW': ‘成都东', ‘CNW': ‘成都南', ‘CDW': ‘成都', ‘CSQ': ‘长沙', ‘CWQ':
‘长沙南',}

接下来我们就动手开始程序的主要代码编写:


   def main():
     date     = input("请输入时间(如2019-01-22):\n")
     from_station = chezhan_code[input("请输入起始站点:\n")]
     to_station  = chezhan_code[input("请输入目的站点:\n")]
     url     = "https://kyfw.12306.cn/otn/leftTicket/queryZ?"
     headers = {
       "User-Agent": "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/63.0.3239.26 Safari/537.36 Core/1.63.5702.400 QQBrowser/10.2.1893.400"
     }
     url=url+"leftTicketDTO.train_date="+date+"&leftTicketDTO.from_station="+from_station+"&leftTicketDTO.to_station="+to_station+"&purpose_codes=ADULT"
     #print(url) 已经检查过生成的URL是正确的
     #request请求获取主页
     r = requests.get(url,headers=headers)
     r.raise_for_status()  #如果发送了一个错误的请求,会抛出异常
     r.encoding = r.apparent_encoding
     showTicket(r.text)

用户输入时间、起始站点、目的站点,然后通过get来请求,然后我们对返回的网页信息进行解析。我们现将上面代码的r.text进行打印,看看我们请求之后,返回了什么样的信息,然后决定我们应该如何解析

教你用python实现12306余票查询

运行结果

这样看着不方便,我们粘贴到记事本中,进行详细的分析:

教你用python实现12306余票查询

请求返回的结果信息

可以与12306显示的信息进行对比,K829是车次,CDW与BJQ是出发地和目的地,10:10是出发时间,06:13是到达时间,44:21是历时时间,20190123为查询的日期,剩下的就是一系列票的各种信息。

下面就是对这些返回的信息进行解析,其实这也是python爬虫的关键,就是解析!!!

我们先把信息转化为json格式,可以看到都是用“|”隔开的,那么我们就用split函数分割出来,下面是主要功能代码:


   def showTicket(html):
     html = json.loads(html)
     table = PrettyTable([" 车次 ","出发车站","到达车站","出发时间","到达时间"," 历时 ","商务座"," 一等座","二等座","高级软卧","软卧","动卧","硬卧","软座","硬座","无座","其他","备注"])
     for i in html['data']['result']:
       name = [
             "station_train_code",
             "from_station_name",
             "to_station_name",
             "start_time",
             "arrive_time",
             "lishi",
             "swz_num",
             "zy_num",
             "ze_num",
             "dw_num",
             "gr_num",
             "rw_num",
             "yw_num",
             "rz_num",
             "yz_num",
             "wz_num",
             "qt_num",
             "note_num"
           ]
       data = {
             "station_train_code": '',
             "from_station_name": '',
             "to_station_name": '',
             "start_time": '',
             "arrive_time": '',
             "lishi": '',
             "swz_num": '',
             "zy_num": '',
             "ze_num": '',
             "dw_num": '',
             "gr_num": '',
             "rw_num": '',
             "yw_num": '',
             "rz_num": '',
             "yz_num": '',
             "wz_num": '',
             "qt_num": '',
             "note_num": ''
           }
       #将各项信息提取并赋值
       item = i.split('|')                 #使用“|”进行分割
       data["station_train_code"] = item[3]        #获取车次信息,在3号位置
       data["from_station_name"]  = item[6]        #始发站信息在6号位置
       data["to_station_name"]   = item[7]        #终点站信息在7号位置
       data["start_time"]     = item[8]        #出发时间在8号位置
       data["arrive_time"]     = item[9]        #抵达时间在9号位置
       data["lishi"]        = item[10]       #经历时间在10号位置
       data["swz_num"]       = item[32] or item[25] #特别注意,商务座在32或25位置
       data["zy_num"]       = item[31]       #一等座信息在31号位置
       data["ze_num"]       = item[30]       #二等座信息在30号位置
       data["gr_num"]       = item[21]       #高级软卧信息在21号位置
       data["rw_num"]       = item[23]       #软卧信息在23号位置
       data["dw_num"]       = item[27]       #动卧信息在27号位置
       data["yw_num"]       = item[28]       #硬卧信息在28号位置
       data["rz_num"]       = item[24]       #软座信息在24号位置
       data["yz_num"]       = item[29]       #硬座信息在29号位置
       data["wz_num"]       = item[26]       #无座信息在26号位置
       data["qt_num"]       = item[22]       #其他信息在22号位置
       data["note_num"]      = item[1]        #备注信息在1号位置
       color = Colored()
       data["note_num"] = color.white(item[1])
       #如果没有信息,那么就用“-”代替
       for pos in name:
         if data[pos] == "":
           data[pos] = "-"
       tickets = []
       cont = []
       cont.append(data)
       for x in cont:
         tmp = []
         for y in name:
           if y == "from_station_name":
             s = color.green(chezhan_names[data["from_station_name"]])
             tmp.append(s)
           elif y == "to_station_name":
             s = color.red(chezhan_names[data["to_station_name"]])
             tmp.append(s)
           elif y == "start_time":
             s = color.green(data["start_time"])
             tmp.append(s)
           elif y == "arrive_time":
             s = color.red(data["arrive_time"])
             tmp.append(s)
           elif y == "station_train_code":
             s = color.yellow(data["station_train_code"])
             tmp.append(s)
           else:
             tmp.append(data[y])
         tickets.append(tmp)
       for ticket in tickets:
         table.add_row(ticket)
     print(table)

那么我们程序就成功啦!!!

教你用python实现12306余票查询

运行结果

但是在编译器里面Prettytable的格子没有对齐,不要担心,我们到终端运行一下脚本,就可以看到很好看的输出啦:

教你用python实现12306余票查询

终端运行结果

教你用python实现12306余票查询

完成!!!下面是完整代码

main.py


   # -*- coding: utf-8 -*-
   import re,requests,datetime,time,json
   from prettytable import PrettyTable
   from colorama import init,Fore
   from stationinfo import chezhan_code,chezhan_names
   init(autoreset=False)
   class Colored(object):
     def yeah(self,s):
       return Fore.LIGHTCYAN_EX + s + Fore.RESET
     def green(self,s):
       return Fore.LIGHTGREEN_EX + s + Fore.RESET
     def yellow(self,s):
       return Fore.LIGHTYELLOW_EX + s + Fore.RESET
     def white(self,s):
       return Fore.LIGHTWHITE_EX + s + Fore.RESET
     def blue(self,s):
       return Fore.LIGHTBLUE_EX + s + Fore.RESET
   def showTicket(html):
     html = json.loads(html)
     table = PrettyTable([" 车次 ","出发车站","到达车站","出发时间","到达时间"," 历时 ","商务座"," 一等座","二等座","高级软卧","软卧","动卧","硬卧","软座","硬座","无座","其他","备注"])
     for i in html['data']['result']:
       name = [
             "station_train_code",
             "from_station_name",
             "to_station_name",
             "start_time",
             "arrive_time",
             "lishi",
             "swz_num",
             "zy_num",
             "ze_num",
             "dw_num",
             "gr_num",
             "rw_num",
             "yw_num",
             "rz_num",
             "yz_num",
             "wz_num",
             "qt_num",
             "note_num"
           ]
       data = {
             "station_train_code": '',
             "from_station_name": '',
             "to_station_name": '',
             "start_time": '',
             "arrive_time": '',
             "lishi": '',
             "swz_num": '',
             "zy_num": '',
             "ze_num": '',
             "dw_num": '',
             "gr_num": '',
             "rw_num": '',
             "yw_num": '',
             "rz_num": '',
             "yz_num": '',
             "wz_num": '',
             "qt_num": '',
             "note_num": ''
           }
       #将各项信息提取并赋值
       item = i.split('|')                 #使用“|”进行分割
       data["station_train_code"] = item[3]        #获取车次信息,在3号位置
       data["from_station_name"]  = item[6]        #始发站信息在6号位置
       data["to_station_name"]   = item[7]        #终点站信息在7号位置
       data["start_time"]     = item[8]        #出发时间在8号位置
       data["arrive_time"]     = item[9]        #抵达时间在9号位置
       data["lishi"]        = item[10]       #经历时间在10号位置
       data["swz_num"]       = item[32] or item[25] #特别注意,商务座在32或25位置
       data["zy_num"]       = item[31]       #一等座信息在31号位置
       data["ze_num"]       = item[30]       #二等座信息在30号位置
       data["gr_num"]       = item[21]       #高级软卧信息在21号位置
       data["rw_num"]       = item[23]       #软卧信息在23号位置
       data["dw_num"]       = item[27]       #动卧信息在27号位置
       data["yw_num"]       = item[28]       #硬卧信息在28号位置
       data["rz_num"]       = item[24]       #软座信息在24号位置
       data["yz_num"]       = item[29]       #硬座信息在29号位置
       data["wz_num"]       = item[26]       #无座信息在26号位置
       data["qt_num"]       = item[22]       #其他信息在22号位置
       data["note_num"]      = item[1]        #备注信息在1号位置
       color = Colored()
       data["note_num"] = color.white(item[1])
       #如果没有信息,那么就用“-”代替
       for pos in name:
         if data[pos] == "":
           data[pos] = "-"
       tickets = []
       cont = []
       cont.append(data)
       for x in cont:
         tmp = []
         for y in name:
           if y == "from_station_name":
             s = color.green(chezhan_names[data["from_station_name"]])
             tmp.append(s)
           elif y == "to_station_name":
             s = color.yeah(chezhan_names[data["to_station_name"]])
             tmp.append(s)
           elif y == "start_time":
             s = color.green(data["start_time"])
             tmp.append(s)
           elif y == "arrive_time":
             s = color.yeah(data["arrive_time"])
             tmp.append(s)
           elif y == "station_train_code":
             s = color.yellow(data["station_train_code"])
             tmp.append(s)
           else:
             tmp.append(data[y])
         tickets.append(tmp)
       for ticket in tickets:
         table.add_row(ticket)
     print(table)
   def main():
     date     = input("请输入时间:\n")
     from_station = chezhan_code[input("请输入起始站点:\n")]
     to_station  = chezhan_code[input("请输入目的站点:\n")]
     url     = "https://kyfw.12306.cn/otn/leftTicket/queryZ?"
     headers = {
       "User-Agent": "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/63.0.3239.26 Safari/537.36 Core/1.63.5702.400 QQBrowser/10.2.1893.400"
     }
   url=url+"leftTicketDTO.train_date="+date+"&leftTicketDTO.from_station="+from_station+"&leftTicketDTO.to_station="+to_station+"&purpose_codes=ADULT"
     #print(url) 已经检查过生成的URL是正确的
     #request请求获取主页
     r = requests.get(url,headers=headers)
     r.raise_for_status()  #如果发送了一个错误的请求,会抛出异常
     r.encoding = r.apparent_encoding
     showTicket(r.text)
     #print(r.text)
   main()

stationinfo.py


   import re,requests
   url = "https://kyfw.12306.cn/otn/resources/js/framework/station_name.js?station_version=1.8971"
   response = requests.get(url,verify=False)
   #将车站的名字和编码进行提取
   chezhan = re.findall(r'([\u4e00-\u9fa5]+)\|([A-Z]+)', response.text)
   chezhan_code = dict(chezhan)
   chezhan_names = dict(zip(chezhan_code.values(),chezhan_code.keys()))
   #print(chezhan_names)

来源:https://blog.csdn.net/weixin_51463905/article/details/117821678

标签:python,12306
0
投稿

猜你喜欢

  • 对SQL Server聚集索引的指示综合描述

    2010-08-31 14:25:00
  • HTML5的革新:结构之美

    2011-04-16 10:57:00
  • Python实现方便使用的级联进度信息实例

    2021-04-22 12:31:18
  • zen coding的dreamweaver插件安装教程

    2010-01-12 13:30:00
  • Python pandas如何向excel添加数据

    2021-07-14 17:37:41
  • sqlserver 动态创建临时表的语句分享

    2012-01-29 17:54:37
  • 在Heroku云平台上部署Python的Django框架的教程

    2022-02-28 08:46:49
  • 使用 XML 模板 (MSSQL手册)

    2008-09-04 14:25:00
  • CVE-2020-15148漏洞分析

    2023-06-13 13:41:18
  • 网页设计细节不能丢

    2007-09-26 11:54:00
  • 超简单使用Python换脸实例

    2021-07-31 17:03:36
  • 在ASP与ASP.NET之间共享对话状态(2)

    2008-09-02 12:21:00
  • 如何实现上下翻页?

    2010-05-24 18:29:00
  • 利用python实现命令行有道词典的方法示例

    2021-03-24 06:32:16
  • 利用JavaScript正则表达式模拟Google Talk的文本处理

    2007-12-04 18:43:00
  • 关于超级链接的一些问题

    2007-12-07 14:00:00
  • python使用KNN算法识别手写数字

    2022-02-20 10:48:23
  • python版本的读写锁操作方法

    2023-04-29 05:18:57
  • python 定时器,轮询定时器的实例

    2023-07-08 04:26:57
  • python 装饰器功能以及函数参数使用介绍

    2022-04-03 05:12:32
  • asp之家 网络编程 m.aspxhome.com