python之12306自动查票

一、导读

本篇文章所采用的技术仅用于学习、研究,任何其他用途请自行承担后果。

12306自动查票使用到的python库主要是splinter,同时也涉及到查票的城市编码,具体的城市编码请在网络上搜索,基本格式如下:

北京北:VAP

北京东:BOP

北京: BJP

北京南:VNP

北京西:BXP

实现的功能包括:
(1)自动打开Google浏览器,进入12306登录页面
(1)命令行手动输入账号、密码、出发、目的地,时间等相关信息,登录验证图片需要手动选择。
(2)自动填充输入,完成查询和页面跳转。

后续可以优化的功能:
(1)从配置文件中读取账号、密码、出发、目的地,时间等相关信息。
(2)从配置文件中获取城市编码
(3)登录验证图片可以使用第三方识别自动选择

二、初识Splinter

1.简介

Splinter 是用 Python 开发的一个开源web自动化测试的工具集。 它可以帮你自动化浏览器的行为,比如浏览 URLs 并和页面进行交互。

Splinter是现有浏览器之上抽象层自动化工具(如 Selenium, PhantomJSzope.testbrowser )。它具有 高级API ,这使得它很容易去编写Web应用程序的自动化测试。

例如, 用Splinter填写一个表单项:

browser.fill('username', 'janedoe')

在Selenium中, 等效代码会是:

elem = browser.find_element.by_name('username')

elem.send_keys('janedoe')

2.安装

命令行下执行以下命令:sudo pip install splinter

3.快速上手

(1)导入Browser类

from splinter.browser import Browser

(2)创建一个实例

指定driver为chrome浏览器,如果你不为 Browser 指定 driver, 那么会默认使用 firefox。

browser = Browser(driver_name='chrome',executable_path='xxx')

其中executable_path为对应浏览器driver的本地目录。

(3)访问百度搜索页面

使用 browser.visit 方法可访问任意网站:

browser.visit('http://baidu.com')

(4)输入搜索关键词

页面加载完毕后,可以在输入框填充字段,大过年的我们就搜索一下2018年新年祝福吧:

browser.fill('wd', '2018年新年祝福')

(5)点击搜索按钮

Splinter 可以通过按钮的css, xpath, id, tag 或 name来识别,百度搜素按钮使用以下来操作:

button = browser.find_by_xpath('//input[@type="submit"]').click()

(6)匹配结果

使用is_text_present查看匹配结果:

if browser.is_text_present('春节'):

    print("找到了")

else:

    print("没找到")

(7)关闭浏览器

结束测试后,我们需要使用 browser.quit 关闭浏览器:

browser**.**quit()

完整代码如下:

# -*- coding: utf-8 -*-

#导入Browser类

from splinter.browser import Browser

#创建一个实例, 指定driver为chrome浏览器,如果你不为 Browser 指定 driver, 那么会默认使用 firefox。

browser = Browser(driver_name='chrome',executable_path='/Users/xxx/Downloads/chromedriver')

browser.visit('http://baidu.com')

browser.fill('wd', '2018年新年祝福')

button = browser.find_by_xpath('//input[@type="submit"]').click()

if browser.is_text_present('春节'):

    print("找到了")

else:

    print("没找到")

browser.quit()

三、12306自动查票
1、流程分析
(1)执行python脚本后,能够自动打开浏览器,进入12306登录页面。因此需要加载浏览器驱动并打开登录页面。

(2)命令行提示用户输入用户名、密码,并等待用户手动在浏览器选择验证码完成登录。

(3)命令行提示用户输入出发地、目的地以及出发时间。

(4)根据输入查询车次信息

2、示例代码
代码均有详细注释,根据上面的流程分析,我们简化为三步。
第一步是加载基本信息,包括浏览器、url等。第二步是输入个人信息登录。第三步是输入查询条件查询车次信息。

# -*- coding: utf-8 -*-

from splinter.browser import Browser
from time import sleep

class TicketsUtil(object):
    
    def __init__(self):
        self.loadBasicInfo()
        
    def loadBasicInfo(self):
        # 登录的url
        self.loginUrl = 'https://kyfw.12306.cn/otn/login/init'
        #登录成功后的url
        self.myUrl = 'https://kyfw.12306.cn/otn/index/initMy12306'
        #余票查询页面
        self.ticketUrl = 'https://kyfw.12306.cn/otn/leftTicket/init'
        # 初始化驱动
        self.driver=Browser(driver_name='chrome',executable_path='/Users/xxx/Downloads/chromedriver')
        # 初始化浏览器窗口大小
        self.driver.driver.set_window_size(1400, 1000)

    def login(self):
        print("开始登录...")
        # 登录
        self.driver.visit(self.loginUrl)
        
        self.username = input("\n请输入用户名,输入按回车...")
        #合法性判断
        while True:   
            if self.username == '':
                self.username = input("\n请输入用户名,输入按回车...")
            else:
                break

        self.password = input("\n请输入密码,输入按回车...")
        #合法性判断
        while True:   
            if self.password == '':
                self.password = input("\n请输入密码,输入按回车...")
            else:
                break

        # 自动填充用户名
        self.driver.fill("loginUserDTO.user_name", self.username)
        # 自动填充密码
        self.driver.fill("userDTO.password", self.password)
            

        print(u"等待验证码,自行输入...")

        # 验证码需要自行输入,程序自旋等待,直到验证码通过,点击登录
        while True:
            if self.driver.url != self.myUrl:
                sleep(1)
            else:
                break
        
        print(u"登录成功...")
                
    def query(self):
        self.source = input("\n请输入出发地(格式为:北京,BJP),输入按回车...")
        #合法性判断
        while True:   
            if self.source == '':
                self.source = input("\n请输入出发地(格式为:北京,BJP),输入按回车...")
            else:
                break
                
        self.destination = input("\n请输入目的地(格式为:深圳,SZQ),输入按回车...")
        while True:   
            if self.destination == '':
                self.destination = input("\n请输入目的地(格式为:深圳,SZQ),输入按回车...")
            else:
                break
                
        self.date = input("\n请输入出发日期(格式为:2018-02-14),输入按回车...")
        while True:   
            if self.date == '':
                self.date = input("\n请输入出发日期,输入按回车...")
            else:
                break
        #转换输入的出发地成"武汉,WHN",再进行编码
        self.source = self.source.encode('unicode_escape').decode("utf-8").replace("\\u", "%u").replace(",", "%2c")
        #转换输入的目的地
        self.destination = self.destination.encode('unicode_escape').decode("utf-8").replace("\\u", "%u").replace(",", "%2c")
                  
        # 加载出发地
        self.driver.cookies.add({"_jc_save_fromStation": self.source})
        # 加载目的地
        self.driver.cookies.add({"_jc_save_toStation": self.destination})
        # 加载出发日
        self.driver.cookies.add({"_jc_save_fromDate": self.date})
        
        # 带着查询条件,重新加载页面
        self.driver.reload()
        # 查询余票
        self.driver.find_by_text(u"查询").click()
        sleep(0.1)
        # 防止超时再次查询余票
        self.driver.find_by_text(u"查询").click()
        
        print('查询成功')

    """入口函数"""
    def start(self):
        self.loadBasicInfo()

        # 登录,自动填充用户名、密码,自旋等待输入验证码,输入完验证码,点登录后,访问 tick_url(余票查询页面)
        self.login()

        # 登录成功,访问余票查询页面
        self.driver.visit(self.ticketUrl)
        
        self.query()

if __name__ == '__main__':
    print(u"===========自动查票开启===========")
    ticketsUtil = TicketsUtil()
    ticketsUtil.start()


©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容