目录

一、效果展示

二、代码详解

1 导入库

2 确定好购票基本信息

3 登录12306

4 模拟滑动滑块

5 处理疫情特殊要求

6 点击购票并填写出发地、目的地、出发时间

7 锁定车票

三、实现监控购买

注意

环境

代码


大家有没有这种感觉,一到国庆、春节这种长假,抢火车票就非常困难?各大互联网公司都推出抢票服务,只要加钱给服务费就可以增加抢到票的几率。有些火车票代售网点和一些加速买票软件,说你只要给100元服务费就可以优先帮忙抢到票。

本文和你一起探索抢票软件背后的原理。

一、效果展示

自动买火车票

在正式进入代码讲解之前,先来看下本文的实现效果。

如果不是为了演示效果,直接在最后确定阶段加一个延时点击确定,应该不到45秒可以锁定一张票,只要在30分钟之内付款即可。

二、代码详解

本小节会详细解锁抢票软件是如何模拟登录网站,进行自动买票的。为了更清晰地给大家展示,部分代码没有写成函数,直接裸代码运行,让需要买票的朋友可以自己应用软件进行购票。

1 导入库

首先导入本文需要加载的库,如果你有些库还没有安装,导致运行代码时报错,可以在Anaconda Prompt中用pip方法安装。

1

2

3

4

5

6

7

8

9

10

11

  1. ###########################

  2. Python学习交流Q群101677771

  3. ###########################

import json

import time

from captcha import *

from selenium import webdriver

from selenium.webdriver.common.by import By

from selenium.webdriver.support import wait

from selenium.webdriver.common.keys import Keys

from selenium.common.exceptions import NoSuchElementException

from selenium.webdriver.common.action_chains import ActionChains

from selenium.webdriver.support import expected_conditions as EC

#导入库

2 确定好购票基本信息

导入库后,在python代码中填写你购票的基本信息。

1

2

3

4

5

6

7

8

9

purpose = 'ADULT'               #购买成人票,如果是学生票,需调整代码

names = ['谢朝阳']              #填写购票人姓名

date = '2021-09-21'             #填写购票日期

start_station = '深圳'          #购票出发站

end_station = '长沙南'          #购票目的站

password = '11234567xyz'        #登录12306的秘密

username ='xiezhaoyang122700'   #登录12306的账号

trains = ['G1004', 'G80', 'G6028', 'G6182', 'G6016' #你想买的班次

#填写基本信息

本文预订的是2021年9月21日从深圳到长沙南的高铁票,你可以根据自己的实际需要进行调整。由于有些班次的时间过早或过晚,买了也很不方便,所以可以在trains中挑选出你满意的班次进行购票。在这里需要提醒大家,我之前在尝试代码时碰到的坑,那就是时间中如果有个位数要在前面填0。比如2021年9月2日,你在填写购票日期date时要写成'2021-09-02',否则在运行代码时日期总是填不进去。

3 登录12306

确定好购票的基本信息后,就可以应用python模拟登录12306了,代码如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

options = webdriver.ChromeOptions()

options.add_argument("--disable-blink-features=AutomationControlled"

browser = webdriver.Chrome(options=options)

browser.maximize_window()

login_url = '中国铁路12306'

#ticket_url = '中国铁路12306'

browser.get(login_url)

time.sleep(0.5)

wait.WebDriverWait(browser, 5).until(EC.element_to_be_clickable((By.CLASS_NAME,'login-hd-account'))).click()

input_name = browser.find_element_by_id('J-userName')

input_pd = browser.find_element_by_id('J-password')

input_name.send_keys(username)

input_pd.send_keys(password)

login = browser.find_element_by_id('J-login')

login.click()

#登录12306

整体思路是: 

1.应用python模拟调用google浏览器;

2.输入12306网址;

3.等网页加载完全后点击账户密码登录;

4.找到账号密码的id,把账户密码信息填充进去;

5.找到登录id,模拟点击登录按钮。

在这一小节中要注意两个点。

一、要在python安装目录中放和google版本匹配的chromedriver,供python调用。

二、要学会找填写账户密码信息的id。

首先,在google浏览器中输入12306登录网址:

中国铁路12306

接着点击账户密码登录,会出现如下界面:

然后点击红框中的三个点,找到更多工具,点击开发者工具,会出现如下界面:

点击红框中的箭头,把鼠标移动到账户框上去,就会出现如下界面:

右边变灰的框里就会出现对应的id,点击账号框,再把鼠标移动到右边变灰的字符上去,点击右键,会出现copy element的选项,复制下来即可。

1

<input type="text" class="input" id="J-userName" placeholder="用户名/邮箱/手机号" style="height: 44px; line-height: 44px; outline: black 0px;" aria-label="请输入用户名/邮箱/手机号" title="请输入用户名/邮箱/手机号">

发现了吗?源代码input_name中要填写的browser.find_element_by_id('J-userName')内容,即为id="J-userName"中的信息。

4 模拟滑动滑块

输入完用户名和密码,点击立即登录后,会出现如下滑块验证要求。

运行如下代码即可拖动滑块进行验证。

1

2

3

4

5

6

7

8

9

10

browser.implicitly_wait(5)

print('=====开始处理滑动验证码=====')

track = [300, 400, 500

for i in track:

    try:

        btn = browser.find_element_by_xpath('//*[@id="nc_1__scale_text"]/span')

        ActionChains(browser).drag_and_drop_by_offset(btn,i,0).perform()

    except:

        time.sleep(2

#拉动滑块验证

其中,browser.implicitly_wait(5)表示隐性等待5秒,track中放的是滑块拉动的距离。

5 处理疫情特殊要求

完成滑块验证要求后,会出现如下疫情特殊要求提示:

用如下代码点击确认即可。

1

2

3

4

browser.implicitly_wait(5)

browser.find_element_by_xpath('/html/body/div[5]/div[2]/div[3]/a').click()

time.sleep(2

#疫情特殊要求

browser.find_element_by_xpath和id的区别是,在右键复制时要copy XPath或copy full XPath。

6 点击购票并填写出发地、目的地、出发时间

接下来是选择买票,并将出发地、目的地、出发时间等信息填写进去。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

browser.find_element_by_xpath('//*[@id="J-chepiao"]/a').click()

browser.find_element_by_xpath('//*[@id="megamenu-3"]/div[1]/ul/li[1]/a').click()

browser.find_element_by_xpath('//*[@id="qd_closeDefaultWarningWindowDialog_id"]').click()

#选择买票

def input_info():

    print('=====开始买票=====')

    from_station = browser.find_element_by_xpath('//*[@id="fromStationText"]')

    from_station.send_keys(Keys.ENTER)

    from_station.send_keys(Keys.CONTROL, 'a')

    from_station.send_keys(start_station, Keys.ENTER)

    browser.implicitly_wait(5)

    to_station = browser.find_element_by_xpath('//*[@id="toStationText"]')

    to_station.send_keys(Keys.ENTER)

    to_station.send_keys(Keys.CONTROL, 'a')

    to_station.send_keys(end_station, Keys.ENTER)

    browser.implicitly_wait(5)

    start_date = browser.find_element_by_xpath('//*[@id="train_date"]')

    start_date.send_keys(Keys.ENTER)

    start_date.send_keys(Keys.CONTROL, 'a')

    start_date.send_keys(Keys.CONTROL, 'x')

    start_date.send_keys(date, Keys.ENTER)

    browser.implicitly_wait(5)

    wait.WebDriverWait(browser, 3).until(EC.element_to_be_clickable((By.ID,'query_ticket'))).click()

input_info()

input_info()

#将出发地、目的地、出发日期填进去

得到结果如下:

这里需要注意的是我调用了两遍input_info函数,因为12306可能采取了一些反爬措施,一遍输入进去后查不出东西,显示为灰色。

7 锁定车票

最后是依次查找trains中的车次是否有票,有的话点击购买锁定车票。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

trList = browser.find_elements_by_xpath(".//tbody[@id='queryLeftTable']/tr[not(@datatran)]")

for tr in trList:

    trainNum = tr.find_element_by_class_name("number").text

    if trainNum in trains:

        leftTicket = tr.find_element_by_xpath(".//td[4]").text

        print('leftTicket', leftTicket)

        if leftTicket == '有' or leftTicket.isdigit():

            orderBtn = tr.find_element_by_class_name("btn72")

            orderBtn.click()

            browser.implicitly_wait(5)

            passengerLabels = browser.find_elements_by_xpath(".//ul[@id='normal_passenger_id']/li/label")

            for passengerLabel in passengerLabels:

                name = passengerLabel.text

                if name in names:

                    passengerLabel.click()

            browser.implicitly_wait(20)

            # 获取提交按钮

            submitBtn = browser.find_element_by_id("submitOrder_id")

            submitBtn.click()

            browser.implicitly_wait(20)

            confirmBtn = browser.find_element_by_id("qr_submit_id")

            confirmBtn.click()

            time.sleep(2)

            browser.implicitly_wait(20)

            confirmBtn = browser.find_element_by_id("qr_submit_id")

            confirmBtn.click()

            break

#依次查找trains中的车次是否有票,有的话点击购买

所以,如果你有特别心仪的车次,可以在trains中放在最前面,依次填写觉得还行的车次。至此,应用python解锁抢票软件背后的原理已讲解完毕,感兴趣的朋友可以自己跟着本文实现一遍。

12306不定期会更新买票界面,所以过一段时间可能之前的代码就要进行一些调整,需要自己弄清里面的原理,才可以以不变应万变。本文的代码没有进行高级的封装,只为大家能更清楚地了解每一步,能在抢票高峰期买到自己心仪的票。

也写得很基础,没有进一步的调优缩短时效,感兴趣的朋友可以自行研究,如有任何疑问可以跟我沟通。

三、实现监控购买

原代码只能实现购票,我对代码进行了修改,增加了监测和抢票成功推送功能。
需要自行输入12306账号、密码,购买车次、时间、出发站、目的站、server酱key(Server酱·Turbo版 (ftqq.com)去申请)

经测试可成功购票,但有如下问题需要

注意

1.登录测试过多可能会导致滑块验证失败,可自行点击刷新,即可自动执行后续代码(懒得写刷新验证了,只要不是多次运行,一般不会有这个问题)

2.main()中,买票buy那一块,因为我是要抢票,等待时间设置为2,如果是监测,建议设置时间长一点,因为有可能被反爬

3.买票提交按钮可能会有bug,于是我在main中增加了一次选择来确保。但无论如何,进入提交页面你都会收到微信提示,为了保险,建议去看一下是否真的提交了,手动提交也不是不可以。

4.多人买票请在buy()函数下names里填入

5.代码中那么多**code,是我用来标记这段代码是否执行成功,如果没有成功就重复执行,如果code=0,代码没成功,再从头执行,gmcode和code感觉有点重复了,但是,管他呢,能运行就行了, 滑稽.jpg

6.我只是个业余程序猿,代码改的丑陋,我尽量写的通俗易懂了,大佬勿喷。

环境

(食用本代码需要一定的基础知识,新手估计挺难得)

  • python3
  • webdriver

以及一些库,我用pycharm,没有的库可以直接导入
还有,我觉得最主要的就是webdriver,自己有的话更好,配置的话可能需要很久时间,也挺麻烦的

代码

# -*- coding:utf-8 -*-
import json
import requests
import time
from captcha import *
from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.support import wait
from selenium.webdriver.common.keys import Keys
from selenium.common.exceptions import NoSuchElementException
from selenium.webdriver.common.action_chains import ActionChains
from selenium.webdriver.support import expected_conditions as EC# 定义一系列code来确保每一步执行成功再进入下一步
global logincode, hkcode, yzcode, xpcode, cpcode, gmcode, code# 初始化
def init_program():options = webdriver.ChromeOptions()options.add_argument("--disable-blink-features=AutomationControlled")browser = webdriver.Chrome(options=options)browser.maximize_window()return browser# 登录12306
def login(browser):global logincodelogincode = 0password = ''  # 登录12306的秘密username = ''  # 登录12306的账号login_url = 'https://kyfw.12306.cn/otn/resources/login.html'# ticket_url = 'https://kyfw.12306.cn/otn/leftTicket/init'try:browser.get(login_url)time.sleep(0.5)wait.WebDriverWait(browser, 5).until(EC.element_to_be_clickable((By.CLASS_NAME, 'login-hd-code'))).click()input_name = browser.find_element_by_id('J-userName')input_pd = browser.find_element_by_id('J-password')input_name.send_keys(username)input_pd.send_keys(password)login = browser.find_element_by_id('J-login')login.click()logincode = 1except Exception as e:logincode = 0print(e)# 拉动滑块验证
def huakuai(browser):global hkcodehkcode = 0try:browser.implicitly_wait(5)print('=====开始处理滑动验证码=====')track = [300, 400, 500]for i in track:try:btn = browser.find_element_by_xpath('//*[@id="nc_1__scale_text"]/span')ActionChains(browser).drag_and_drop_by_offset(btn, i, 0).perform()hkcode = 1except:time.sleep(2)except Exception as e:hkcode = 0print(e)# 疫情特殊要求
def yiqingyaoqiu(browser):global yzcodeyzcode = 0try:browser.implicitly_wait(5)try:browser.find_element_by_xpath('/html/body/div[4]/div[2]/div[3]/a').click()yzcode = 1except:try:browser.find_element_by_xpath('/html/body/div[2]/div[7]/div[2]/div[3]/a').click()yzcode = 1except:yzcode = 0finally:time.sleep(2)except Exception as e:yzcode = 0print(e)# 进入买票页面
def enterbuy(browser):global xpcodexpcode = 0try:browser.find_element_by_xpath('//*[@id="J-chepiao"]/a').click()browser.find_element_by_xpath('//*[@id="megamenu-3"]/div[1]/ul/li[1]/a').click()browser.find_element_by_xpath('//*[@id="qd_closeDefaultWarningWindowDialog_id"]').click()xpcode = 1except Exception as e:print(e)xpcode = 0# 将出发地、目的地、出发日期填进去
def input_info(browser):global cpcodecpcode = 0date = '2022-01-24'  # 填写购票日期start_station = ''  # 购票出发站,例如南京南end_station = ''  # 购票目的站try:print('=====开始买票=====')from_station = browser.find_element_by_xpath('//*[@id="fromStationText"]')from_station.send_keys(Keys.ENTER)from_station.send_keys(Keys.CONTROL, 'a')from_station.send_keys(start_station, Keys.ENTER)browser.implicitly_wait(5)to_station = browser.find_element_by_xpath('//*[@id="toStationText"]')to_station.send_keys(Keys.ENTER)to_station.send_keys(Keys.CONTROL, 'a')to_station.send_keys(end_station, Keys.ENTER)browser.implicitly_wait(5)start_date = browser.find_element_by_xpath('//*[@id="train_date"]')start_date.send_keys(Keys.ENTER)start_date.send_keys(Keys.CONTROL, 'a')start_date.send_keys(Keys.CONTROL, 'x')start_date.send_keys(date, Keys.ENTER)browser.implicitly_wait(5)wait.WebDriverWait(browser, 3).until(EC.element_to_be_clickable((By.ID, 'query_ticket'))).click()cpcode = 1except Exception as e:print(e)cpcode = 0# 依次查找trains中的车次是否有票,有的话点击购买
def buy(browser):global gmcode, codegmcode = 0code = 0purpose = 'ADULT'  # 购买成人票,如果是学生票,需调整代码names = ['']  # 填写购票人姓名,需要在你的乘车人管理里有的trains = []  # 你想买的班次,例如'D666',  'G666'browser.implicitly_wait(5)try:trList = browser.find_elements_by_xpath(".//tbody[@id='queryLeftTable']/tr[not(@datatran)]")for tr in trList:trainNum = tr.find_element_by_class_name("number").textif trainNum in trains:leftTicket = tr.find_element_by_xpath(".//td[4]").textprint('leftTicket', leftTicket)if leftTicket == '有' or leftTicket.isdigit():orderBtn = tr.find_element_by_class_name("btn72")orderBtn.click()browser.implicitly_wait(5)passengerLabels = browser.find_elements_by_xpath(".//ul[@id='normal_passenger_id']/li/label")for passengerLabel in passengerLabels:name = passengerLabel.textif name in names:passengerLabel.click()browser.implicitly_wait(20)# 获取提交按钮submitBtn = browser.find_element_by_id("submitOrder_id")submitBtn.click()browser.implicitly_wait(20)confirmBtn = browser.find_element_by_id("qr_submit_id")confirmBtn.click()time.sleep(2)browser.implicitly_wait(20)confirmBtn = browser.find_element_by_id("qr_submit_id")confirmBtn.click()code = 1gmcode = 1breakexcept Exception as e:print(e)gmcode = 0def tuisong():api = "https://sctapi.ftqq.com/*****.send" #*****替换成你的微信server酱的key,可以实现购票成功推送,然后你就自己去12306付款title = '购买成功'data = {"text": title}req = requests.post(api, data=data)if __name__ == "__main__":global logincode, yzcode, hkcode, xpcode, cpcode, gmcode, codecode = 0logincode = 0yzcode = 0hkcode = 0xpcode = 0cpcode = 0gmcode = 0browser = init_program()while code == 0:while logincode == 0:login(browser)print('logincode:', logincode)while hkcode == 0:huakuai(browser)print('hkcode:', hkcode)while yzcode == 0:yiqingyaoqiu(browser)print('yzcode:', yzcode)while xpcode == 0:enterbuy(browser)print('xpcode:', xpcode)while cpcode == 0:input_info(browser)input_info(browser)#经测试,一次有可能不成功,我直接两次提交print('cpcode:', cpcode)while gmcode == 0:buy(browser)print('gmcode:', gmcode)print('code:', code)if gmcode == 0:browser.refresh()time.sleep(2)browser.find_element_by_xpath('//*[@id="qd_closeDefaultWarningWindowDialog_id"]').click()input_info(browser)input_info(browser)else:try:print('tijiao')confirmBtn = browser.find_element_by_id("qr_submit_id")browser.implicitly_wait(20)time.sleep(3)confirmBtn.click()except:passif code == 1:tuisong()break

要过年了,最新的12306实时监控抢购车票软件已经写好了,这多亏我Python学的好相关推荐

  1. python实时监控文件大小_python实现实时监控文件的方法

    在业务稳定性要求比较高的情况下,运维为能及时发现问题,有时需要对应用程序的日志进行实时分析,当符合某个条件时就立刻报警,而不是被动等待出问题后去解决,比如要监控nginx的$request_time和 ...

  2. python 远程桌面爆破,Python安全运维第一弹 --实时监控远程桌面连接

    大部分安全工作者都会选择学习python这门语言,常用的内建模块以及大量的第三方模块能够帮助我们快速的实现自己的目的.在日常的安全运维中,很多时候我们会碰到避免安装第三方软件或是第三方软件无法契合自己 ...

  3. Python——实现网课实时监控自动签到,打卡

    3.17日补充 生成uuid(通用唯一识别码),简单来说就是一串随机的字符串,网上有很多在线接口可以生成,大牛也可以自己写程序实现 7e60d79bb2164df8bb2242100979a2e1 首 ...

  4. java对接海康威视SDK(win64、linux64),处理播放实时流转码,按时间回放功能,附海康威视最新3.0摄像头监控web端实例+插件

    java对接海康威视SDK(win64.linux64),处理播放实时流转码,按时间回放功能 准备工作 遇到的坑 调用步骤 参数配置 DeviceEnums 初始化sdk 注册 大华的sdk叫登陆海康 ...

  5. 实时监控:基于流计算 Oceanus ( Flink ) 实现系统和应用级实时监控

    作者:吴云涛,腾讯 CSIG 高级工程师 本文描述了如何使用腾讯云大数据组件来完成实时监控系统的设计和实现,通过实时采集并分析云服务器(CVM)及其 App 应用的 CPU和内存等资源消耗数据,以短信 ...

  6. 基于邮件系统的远程实时监控系统的实现 Python版

    人生苦短,我用Python~ 界内的Python宣传标语,对Python而言,这是种标榜,实际上,Python确实是当下最好用的开发语言之一. 在相继学习了C++/C#/Java之后,接触Python ...

  7. 大屏实时监控-2019年CSDN博客之星年度总评选(2019-02-07 13:47)

    监控地址:http://anymk.com:8888/ 2019-02-07 13:47 2019-02-01 20:09 2019-01-28 15:17 2019-01-25 15:01 大屏实时 ...

  8. 【Flink】Flink 任务实时监控

    1.概述 转载:Flink 任务实时监控最佳实践 我们都知道 Flink 任务是一个 7*24 小时不停运行的任务,所以对于任务的实时监控就显得尤为重要,因为任务运行的状态对于我们来说是一个黑盒,比如 ...

  9. Jmeter系列之Jmeter+Grafana+InfluxDB实时监控

    VOL 167 20 2020-10 今天距2021年72天 这是ITester软件测试小栈第167次推文 点击上方蓝字"ITester软件测试小栈"关注我,每周一.三.五早上 0 ...

最新文章

  1. Symfony学习笔记
  2. CentOS-7.0的核心安装与配置的补全
  3. DHL出台货运新方案
  4. php post 获取xml,php 获取post的xml数据并解析示例
  5. 简单几招提速 Kotlin Kapt编译
  6. 【转载】【凯子哥带你学Framework】Activity界面显示全解析(下)
  7. vue基础之组件(创建,data,切换,父子以及同级之间的传值,插槽solt)
  8. win10下如何安装.NetFrame3.5框架
  9. php操作sqlserver视频教程,SQLServer数据库基础视频教程下载
  10. 一次内存泄露的分析及总结
  11. 【python 数据分析】不同情况下的t检验、Wilcoxon符号秩检验、Wilcoxon秩和检验、卡方检验、Fisher检验
  12. DOA算法3:Matrix Pencil
  13. php魔方阵,利用C语言玩转魔方阵实例教程
  14. 小米路由器Mesh,信号有多牛?
  15. 不等距双杆模型_电磁感应的双杆模型——不等距无拉力
  16. POJ Curling 2.0
  17. uboot中 使用i2c
  18. Duilib介绍-2
  19. CUDA加速计算矩阵乘法进阶玩法(共享内存)
  20. 耐高温滑环在特殊环境作业时需要具备哪些条件

热门文章

  1. 装修流程、步骤详解图
  2. 社群SOP:一个工具让社群运营事半功倍
  3. aliyun基础操作
  4. 生活自有苦乐,苦中作乐,乐中取苦
  5. 【玩转SpringBoot】翻身做主人,一统web服务器
  6. MyEclipse破解 注册机“System id无法生成” 解决方法:出现com.jniwrapper.util.ProcessorInfo
  7. java 8精简jre_精简压缩jre
  8. 热血江湖服务器维护时间,《热血江湖》8月12日更新维护公告 新增灵兽宝珠(红龙)...
  9. 论文精读《BEVDet: High-Performance Multi-Camera 3D Object Detection in Bird-Eye-View》
  10. WebApi--------找到了与该请求匹配的多个操作问题解决