百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

python 12306自动抢票(python自动化抢票)

off999 2024-09-27 13:46 20 浏览 0 评论

又到了一年一度的春运,又要开始抢票了,因为并不是很相信XC、FZ等预约抢票,就想着自己写个脚本,但其实网上已经有很多人写了抢票程序,我为什么还要重写呢,因为12306的网页源代码是有变化的,很多人以前写的已经用不了了,也许我这个程序过一段时间也用不了了。

准备工作

1.使用writeStation.py爬取站点名和对应的简称,写入station.txt文件。以下是writeStation.py的代码和station.txt的部分截图。

#encoding=utf-8
import requests

url = 'https://kyfw.12306.cn/otn/resources/js/framework/station_name.js?station_version=1.9161'
res = requests.get(url)
pageContent = res.content.decode('utf-8')
pageList = pageContent.split('|')
print(pageList)
with open('station.txt', 'w', encoding='utf-8') as f:
    for i in range(len(pageList)):
        if (i - 1) % 5 == 0:
            f.write(pageList[i] + ' ')
        if (i - 2) % 5 == 0:
            f.write(pageList[i] + '\n')
1234567891011121314


2. 比如我们想在2月10日,从深圳去成都,就在携程上这样查,然后我们可以查到,在1月12日的10点会放深圳东站到成都东站,车号为Z332的票。



3.邮箱准备:邮箱用于列车有票的时候,可以给你发邮件通知。需要邮箱号以及该号的客户端授权码。
4.在12306APP上的乘车人里需要有乘车人的名字,比如你叫“徐晓峰”,那你就要把你的个人信息添加到12306→我的→徐晓峰→乘车人→添加“需要乘车人”的信息。
5.修改主程序catchTic.py中的初始化信息,如图:


6.运行程序,因为代码中的一些设置,比如1月12日的10点放票,那么就在1月12日的9点以后再启动程序catchTic.py都可以。

代码

下面是catchTic.py,自认为注释比较详细 (手动狗头)

from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait
import time
import smtplib
from email.mime.text import MIMEText
from email.header import Header
from fake_useragent import UserAgent
import traceback
import random


browser = webdriver.Chrome()

#  读取station.txt文件,找到地点对应的简称
def cityNToJ(cityname):
    with open("station.txt", "r", encoding="utf-8") as f:
        cityDic = {}
        for line in f.readlines():
            cityN = line.split(' ')[0]
            cityJ = line.split(' ')[1][:-1]
            cityDic[cityN] = cityN + ',' + cityJ
    # print(cityDic[cityname])
    return cityDic[cityname]

# 邮件通知
def send_mail(notestr):  # 邮箱通知
    msg = MIMEText(notestr, 'plain', 'utf-8')
    subject = '抢票结果通知'
    msg['Subject'] = Header(subject, 'utf-8')
    msg['From'] = 'Tomm<发件人邮箱>'
    msg['To'] = "收件人邮箱"

    # 输入Email地址和口令:
    from_addr = '508607592@qq.com'
    password = 'ietjnxeiybmmbiib'  # 不是登录密码,而是客户端授权码
    # 输入SMTP服务器地址:
    smtp_server = 'smtp.qq.com'
    # 输入收件人地址:
    to_addr = '785251591@qq.com'

    server = smtplib.SMTP()  # SMTP协议默认端口是25
    server.connect(smtp_server, 25)
    server.set_debuglevel(1)
    server.login(from_addr, password)
    server.sendmail(from_addr, to_addr, msg.as_string())
    server.quit()

# 完成提交页面的选择乘客、选择席别、提交订单的功能
def submit_order(name, seatType):

    time.sleep(0.5)
    browser.find_element_by_id("quickQueryPassenger_id").click()
    browser.find_element_by_id("quickQueryPassenger_id").send_keys(name)
    wait = WebDriverWait(browser, 10)
    wait.until(EC.presence_of_all_elements_located((By.XPATH, '//*[@id="normal_passenger_id"]/li[1]/label')))
    browser.find_element_by_xpath('//*[@id="normal_passenger_id"]/li[1]/label').click()

    # 选择席别
    seatDic = {"ZE": "o", "ZY": "M", "SWZ": "9",
               "RW": "4", "YW": "3", "YZ": "1", "GR": "6"}

    if seatType != "ZE":
        browser.find_element_by_xpath('//*[@id="seatType_1"]').click()
        seatxpath = "//select[@id='seatType_1']/option[@value='{}']".format(seatDic[seatType])
        browser.find_element_by_xpath(seatxpath).click()

    # 提交订单
    time.sleep(0.1)
    browser.find_element_by_xpath('//div[@class="content"]//div[@class="lay-btn"]/a[@id="submitOrder_id"]').click()

    # 核对信息
    time.sleep(2)
    # 提交订单的‘确认’(每天只能确认提交三次)
    browser.find_element_by_link_text('确认').click()
    print("抢到票的时间:", time.strftime("%Y-%m-%d %H:%M:%S", time.localtime()))


def refresh_order():

    # 初始化一些个人信息
    name = r'徐晓峰'  # (手动填写)填写乘车人的名字
    fs = '深圳东'  # (手动填写)出发地
    ts = '成都东'  # (手动填写)目的地
    date = '2020-02-10'  # (手动填写)乘车时间 例:'0000-00-00'
    garbTime = '10:00:00'  # (手动填写)发票时间 例:'00:00:00'
    trainNum = 'Z332'  # (手动填写)列车号
    seatType = 'YW'  # (手动填写)座位类型
    """
    商务座:'SWZ'   软卧:'RW'   高级软卧:'GR'
    一等座:'ZY'    硬卧:'YW'
    二等座:'ZE'    硬座:'YZ'
    """
    seatDic = {'SWZ': '商务座',  'ZY': '一等座', 'ZE': '二等座',
               'RW': '软卧', 'YW': '硬卧', 'YZ': '硬座', 'GR': '高级软卧'}
    fs = cityNToJ(fs)
    ts = cityNToJ(ts)
    flag = 'N,N,Y'  # 不用修改
    linktypeid = 'dc'  # 不用修改
    startUrl = 'https://kyfw.12306.cn/otn/leftTicket/init?'
    url = (startUrl + 'linktypeid={0}' + '&fs={1}' + '&ts={2}' + '&date={3}' + '&flag={4}').format(linktypeid, fs, ts,
                                                                                                   date, flag)

    ua = UserAgent()
    try:
        options = webdriver.ChromeOptions()
        useragent = ua.random
        headers = {'User-Agent': useragent}
        options.add_argument(headers)  # 修改请求头
        browser.get(url)
        # time.sleep(5)
        js = 'window.open("https://kyfw.12306.cn/otn/resources/login.html");'
        browser.execute_script(js)  # 新打开上一行代码的网页
        handles = browser.window_handles  # 获取当前窗口句柄集合(列表类型)
        # 暂停30秒去登录(手慢的话,可以设置60秒)
        time.sleep(30)
        # WebDriverWait(browser.switch_to.window(handles[1]), 1000).until(EC.url_changes('https://kyfw.12306.cn/otn/view/index.html'))
        # 然后切换到抢票界面
        browser.switch_to.window(handles[0])

        time.sleep(3)
        browser.refresh()
        click_query = browser.find_element_by_css_selector('.content.content-lg .sear-box.quick-sear-box.sear-box-lg '
                                                           '.quick-s .btn-area a')
        # index用于统计抢票次数
        index = 0
        #  等待到起售前4秒左右
        garbT = time.strptime(garbTime, "%H:%M:%S")
        if garbT.tm_min == 0:
            garbTime = str(garbT.tm_hour - 1) + ":" + "59" + ":" + "59"
        else:
            garbTime = str(garbT.tm_hour) + ":" + str(garbT.tm_min - 1) + ":" + "59"
        garbT_new = time.strptime(garbTime, "%H:%M:%S")
        while True:
            now = time.localtime()
            if garbT_new.tm_min == now.tm_min and garbT_new.tm_sec - now.tm_sec <= 3:
                print(garbT_new.tm_min, now.tm_min)
                break
            else:
                continue
        # 开始刷新页面,等待目标列车的对应座位号有票出现
        while True:
            browser.execute_script("arguments[0].click();", click_query)
            try:
                WebDriverWait(browser, random.randint(1, 5)).until(EC.presence_of_all_elements_located((By.XPATH, '//*[@id="queryLeftTable"]')))
            except:
                continue
            # time.sleep(5)
            startT = time.time()
            while True:
                # 点击查询,如果点击查询后,列车号一分钟内都找不到
                try:
                    browser.execute_script("arguments[0].click();", click_query)
                    WebDriverWait(browser, random.randint(2, 5)).until(EC.presence_of_all_elements_located((By.XPATH, "//tr[@datatran='{}']".format(trainNum))))
                    break
                except:
                    endT = time.time()
                    if endT - startT <= 60:
                        continue
                    else:
                        print("没有找到此列车!")
                        break
            train_xId = browser.find_element_by_xpath("//tr[@datatran='{}']".format(trainNum)).get_attribute('id')[6:]
            print("trainID:", train_xId)
            trainId = train_xId[:12]
            querySeatType = browser.find_element_by_xpath('//*[@id="{0}_{1}"]'.format(seatType, trainId)).text
            print("querySeatType:", querySeatType)

            if querySeatType != r'无' and querySeatType != r'候补' and querySeatType != '--':
                # 点击‘预订’
                try:
                    browser.find_element_by_xpath('//*[@id="ticket_{}"]/td[13]/a'.format(train_xId)).click()
                    print("成功点击‘预定’...")
                except:
                    continue
                noteStr = "有{},及时查看".format(seatDic[seatType])
                send_mail(noteStr)
                # send_message(noteStr)
                try:
                    submit_order(name, seatType)
                except Exception as e:
                    print("提交订单阶段失败")
                    print('提交订单阶段错误:', e)
                    # browser.back()
                    time.sleep(2)
                    continue
                    # send_message("有余票但是提交订单失败")
                finally:
                    pass
                index = index + 1
                print("第" + str(index) + "次抢票结果: 成功")
                break
            index = index+1
            print("第" + str(index) + "次抢票结果: 失败")

    except:
        print("Something is wrong")
        traceback.print_exc()
        # send_message("程序运行出错")


if __name__ == "__main__":
    refresh_order()

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205

可能出现的报错

  1. 访问12306次数过多,页面出现了有黄色三角的黑色感叹号,解决办法,自己手动用浏览器搜一下12306,在12306里面手动访问一下。
  2. 有些选座类型,比如软卧,12306发的比较少,可能已经有人通过XC等软件预约完了,这个脚本可能没抢过携程,那么刷出来就是“候补”了。一般选“硬卧”或“二等座”都会有票的,这应该也满足了大部分人的需求。当然你也可以自己修改一下代码,让它可以优先判定有没有“软卧”,没有的话,再选“硬卧”。
  3. 可能还有一些报错是我没有发现的,有朋友发现的话,也可以留言或者私信。

相关推荐

编写更多 pythonic 代码(十三)——Python类型检查

一、概述在本文中,您将了解Python类型检查。传统上,类型由Python解释器以灵活但隐式的方式处理。最新版本的Python允许您指定显式类型提示,这些提示可由不同的工具使用,以帮助您更...

[827]ScalersTalk成长会Python小组第11周学习笔记

Scalers点评:在2015年,ScalersTalk成长会完成Python小组完成了《Python核心编程》第1轮的学习。到2016年,我们开始第二轮的学习,并且将重点放在章节的习题上。Pytho...

用 Python 画一颗会跳动的爱心:代码里的浪漫仪式感

在编程的世界里,代码不仅是逻辑的组合,也能成为表达情感的载体。今天我们就来聊聊如何用Python绘制一颗「会跳动的爱心」,让技术宅也能用代码传递浪漫。无论是写给爱人、朋友,还是单纯记录编程乐趣,这...

Python面向对象编程(OOP)实践教程

一、OOP理论基础1.面向对象编程概述面向对象编程(Object-OrientedProgramming,OOP)是一种编程范式,它使用"对象"来设计应用程序和软件。OOP的核心...

如何在 Python 中制作 GIF(python做gif)

在数据分析中使用GIF并发现其严肃的一面照片由GregRakozy在Unsplash上拍摄感谢社交媒体,您可能已经对GIF非常熟悉。在短短的几帧中,他们传达了非常具体的反应,只有图片才能传达...

Python用内置模块来构建REST服务、RPC服务

1写在前面和小伙伴们分享一些Python网络编程的一些笔记,博文为《PythonCookbook》读书后笔记整理博文涉及内容包括:TCP/UDP服务构建不使用框架创建一个REST风格的HTTP...

第七章:Python面向对象编程(python面向对象六大原则)

7.1类与对象基础7.1.1理论知识面向对象编程(OOP)是一种编程范式,它将数据(属性)和操作数据的函数(方法)封装在一起,形成一个称为类(Class)的结构。类是对象(Object)的蓝图,对...

30天学会Python编程:8. Python面向对象编程

8.1OOP基础概念8.1.1面向对象三大特性8.1.2类与对象关系核心概念:类(Class):对象的蓝图/模板对象(Object):类的具体实例属性(Attribute):对象的状态/数据方法...

RPython GC 对象分配速度大揭秘(废土种田,分配的对象超给力)

最近,对RPythonGC的对象分配速度产生了浓厚的兴趣。于是编写了一个小型的RPython基准测试程序,试图探究它对象分配的大致速度。初步测试与问题发现最初的设想是通过一个紧密循环来分配实...

30天学会Python编程:2. Python基础语法结构

2.1代码结构与缩进规则定义与原理Python使用缩进作为代码块的分界符,这是Python最显著的特征之一。不同于其他语言使用大括号{},Python强制使用缩进来表示代码层次结构。特性与规范缩进量...

Python 类和方法(python类的方法与普通的方法)

Python类和方法Python类创建、属性和方法具体是如何体现的,代码中如何设计,请继续看下去。蟒蛇类解释在Python中使用OOP?什么是Python类?Python类创建Pyt...

动态类型是如何一步步拖慢你的python程序的

杂谈人人都知道python慢,这都变成了人尽皆知的事情了,但你知道具体是什么拖慢了python的运行吗?动态类型肯定要算一个!动态类型,能够提高开发效率,能够让我们更加专注逻辑开发,使得编程更加灵活。...

用Python让图表动起来,居然这么简单

我好像看到这个emoji:动起来了!编译:佑铭参考:https://towardsdatascience.com/how-to-create-animated-graphs-in-python-bb6...

Python类型提示工程实践:提升代码质量的静态验证方案

根据GitHub年度开发者调查报告,采用类型提示的Python项目维护成本降低42%,代码审查效率提升35%。本文通过9个生产案例,解析类型系统在工程实践中的应用,覆盖API设计、数据校验、IDE辅助...

Python:深度剖析实例方法、类方法和静态方法的区别

在Python中,类方法(classmethod)、实例方法(instancemethod)和静态方法(staticmethod)是三种不同类型的函数,它们在使用方式和功能上有一些重要的区别。理...

取消回复欢迎 发表评论: