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
可能出现的报错
- 访问12306次数过多,页面出现了有黄色三角的黑色感叹号,解决办法,自己手动用浏览器搜一下12306,在12306里面手动访问一下。
- 有些选座类型,比如软卧,12306发的比较少,可能已经有人通过XC等软件预约完了,这个脚本可能没抢过携程,那么刷出来就是“候补”了。一般选“硬卧”或“二等座”都会有票的,这应该也满足了大部分人的需求。当然你也可以自己修改一下代码,让它可以优先判定有没有“软卧”,没有的话,再选“硬卧”。
- 可能还有一些报错是我没有发现的,有朋友发现的话,也可以留言或者私信。
相关推荐
- 编写更多 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)是三种不同类型的函数,它们在使用方式和功能上有一些重要的区别。理...
你 发表评论:
欢迎- 一周热门
- 最近发表
-
- 编写更多 pythonic 代码(十三)——Python类型检查
- [827]ScalersTalk成长会Python小组第11周学习笔记
- 用 Python 画一颗会跳动的爱心:代码里的浪漫仪式感
- Python面向对象编程(OOP)实践教程
- 如何在 Python 中制作 GIF(python做gif)
- Python用内置模块来构建REST服务、RPC服务
- 第七章:Python面向对象编程(python面向对象六大原则)
- 30天学会Python编程:8. Python面向对象编程
- RPython GC 对象分配速度大揭秘(废土种田,分配的对象超给力)
- 30天学会Python编程:2. Python基础语法结构
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python自定义函数 (53)
- python进度条 (67)
- python吧 (67)
- python字典遍历 (54)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python串口编程 (60)
- python读取文件夹下所有文件 (59)
- java调用python脚本 (56)
- python操作mysql数据库 (66)
- python字典增加键值对 (53)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python人脸识别 (54)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)