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

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

off999 2024-09-27 13:46 31 浏览 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. 可能还有一些报错是我没有发现的,有朋友发现的话,也可以留言或者私信。

相关推荐

erp管理软件(erp管理软件免费版)
erp管理软件(erp管理软件免费版)

用友的ERP应该说从3万-3000都有ERP的价格随着企业的规模不同,价格也是不一样的。因为企业规模不同产生的效果也是不一样的,所以用友的产品分T1/T3/T6/U8/U9/NC3万小企业做个财务业务一体化还是勉强可以做的。只...

2025-11-11 09:03 off999

笔记本启动黑屏怎么回事(笔记本启动黑屏没反应)

笔记本开机后黑屏最常遇到的一种情况:因随便点击垃圾网站而导致电脑中脑或受到木马的侵袭,致使电脑系统瘫痪,解决的办法就是重装电脑系统,装好系统后安装查毒软件,定期对电脑进行杀毒全盘扫描,然后平时尽量不要...

win7系统怎么开wifi热点(win7如何开wifi热点)

 1、首先确认你的无线网卡开启。在开始菜单中依次找到“所有程序”--“附件”--“命令提示符”,右键“以管理员身份运行”;   2、在“命令提示符”里输入“netshwlans...

无线路由桥接设置方法(无线路由器无线桥接设置)

1、首先在电脑上输入并登录第一台路由器的IP地址。2、进入路由器管理界面之后,点击“无线设置”,然后点击基本设置中设置“SSID号”,接着点击“信道”,然后设置固定信号道。3、返回无线设置菜单栏,选择...

win10企业版激活命令(win10企业版cmd激活命令)

关于这个问题,Windows10企业版可以通过以下方法进行激活:1.使用企业版密钥激活:如果你已经有了Windows10企业版的密钥,可以在“设置”中的“更新和安全”中选择“激活”来输入密钥进行...

如何恢复备份数据(备份的数据怎么恢复到手机上)
如何恢复备份数据(备份的数据怎么恢复到手机上)

把备份删了的话,一键还原目前是用不了的。现在唯一的办法,是从网上下载一个数据恢复类的软件,只要的备份还没有被别的软件覆盖,是应该可以数据恢复回来的。不能保证百分之百得成功,但是恢复几率还很大的,你可以试试,操作方法首先点击手机“设置”。然后...

2025-11-11 06:51 off999

笔记本无线网卡怎么使用(笔记本无线网卡怎么使用教程)

笔记本无线网卡设置;第一:你要确定你的本本是否有无线上网功能,如果没有就得加个无线网卡;第二:有的话就打开无线网络接受开关;第三:程序设置主要就是在网上邻居的属性里,打开无线上网打开“网上邻居”的“属...

鲁大师电脑版官方下载(鲁大师电脑版官方下载安装)

因为鲁大师是跑分软件,它会拖慢电脑的运行速度,还会占据大量的内存,如果你的电脑配置不是太好的话,装了鲁大师只会雪上加霜,非但得不到任何优化作用,还会拖慢电脑的启动速度,造成不必要的损耗。玩游戏都会卡顿...

win10怎么开机进入安全模式(win10开机怎么进安全模式怎么进)

进入Windows10安全模式有以下几种方法:方法一:使用开机高级选项1.在按下电源开机键后,持续按住F8键,直到你进入启动选项页面;2.从菜单中选择“安全模式”。方法二:使用系统配置1...

华硕电脑怎么重新安装系统(华硕电脑怎么重新安装系统教程)

第一步:备份重要数据重装系统前,务必先备份重要的个人数据。你可以将数据保存在外部存储设备上,或者使用云存储服务,确保数据安全可靠。第二步:下载系统镜像为了重装系统,你需要下载华硕笔记本电脑适用的操作系...

电脑显示此windows副本不是正版

1、第一步在电脑搜索框搜索命令提示符,鼠标右键以管理员的身份运行,2、第二步以管理员身份进去命令提示符之后输入"SLMGR-REARM",3、第三步按回车键可以看到命令已经成功重启一下...

电脑怎么复制粘贴按键(电脑复制粘贴按键是哪个)

电脑键盘上的粘贴键是:Ctrl+V按键。具体操作:1、以在excel表格中进行复制粘贴操作为例,首先选中需要复制粘贴操作的单元格。2、然后按下键盘上的“Ctrl+C”按键执行复制操作。3、然后将鼠标单...

笔记本黑屏但还在运行(笔记本电脑黑屏但运行)

具体修复方法:1、直接按下电脑机箱上的启动键让电脑重启,等待重新正常进入系统中。然后打开电脑系统盘,右键点击c盘进入属性设置面板中;2、在硬盘的属性设置中切换到工具标签;3、在查错选项中点击检查错误按...

从u盘启动怎么弄bios(u盘怎么在bios启动)

1、开启电脑,在电脑出现开机画面的时候连续按下“Esc”键进入BIOS设置;(部分电脑可能会是Delete、F2或F6)2、进入BIOSFEATURESSETUP中,将Boot(启动顺序)设定为U...

两台电脑怎么共享文件夹(如何把电脑c盘的存储移到d盘)

一、QQ共享简单易行既然使用QQ直接传递文件行不通,那么不妨试试使用QQ的文件共享功能。1.共享文件点击QQ面板的“菜单”→“工具”→“共享文件”命令。在打开的共享文件窗口中,单击“新建共享”按钮,...

取消回复欢迎 发表评论: