python无需验证码免登录12306抢票 --selenium(2)

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档

@[TOC](python无需验证码免登录12306抢票 --selenium(2))


前言

提示:这里可以添加本文要记录的大概内容:

就在刚刚我抢的票:2025年1月8日 上午9.00多 抢到了哈哈哈哈— 其实还是有用的我是在 8:59:56运行程序的

在这里插入图片描述
在这里插入图片描述

上一篇帖子,我们已经了解了如何用python自动登录12306实现自动抢票,现在我们来一个进阶的版本,实现cookie免登录,这样可绕过验证码实现自动抢票。同时包括环境如何配置,也在这个帖子里面。

上一篇帖子的链接:https://blog.csdn.net/xaing1314/article/details/144868504

配置selenium自动化的帖子链接:https://blog.csdn.net/xaing1314/article/details/144869489?spm=1001.2014.3001.5502

需求:上此的帖子,需要自动登录,并且需要手机验证码,在真实抢票中太过繁琐,所有我自己也在思索,找了帖子和B站,终于实现了用cookie技术自动免登录12306。在今天上午9.00 (2025-01-07)抢票(2025-01-21)成功。

如何使用:1.比如9.00正式抢票,可以在8.53左右二维码扫描登录,当然是要配合下方程序,这样9.00可以正式抢票无需登录,cookie也不会失效。2.直接在8.59.58左右运行此程序,即可。

今日实战:今天运行其实是抢到了,但是他一直在等待,也就是页面一直停在哪里,其实那个时候你可以用手点击操作,因为我们是进来了,不需要去管程序的对错。也就是说,程序帮你操作到哪一步,如果卡住了,直接上手。这也让我有想做一个多线程的想法,将多线程加入这个程序中。

在这里插入图片描述

提示:以下是本篇文章正文内容,下面案例可供参考

一、cookie免登录

1.cookie的讲解: Cookie也被称为小型文本文件,是由服务器发送到用户浏览器并保存在用户计算机或移动设备上的一种数据形式。以下是对Cookie的详细解释:

定义与工作原理定义:Cookie是网站在用户浏览器中创建和存储的小型文本文件,用于辨别用户身份和存储用户相关信息。工作原理:当用户访问某个网站时,服务器会在用户的浏览器中创建一个Cookie,并将其存储在用户的计算机或移动设备上。当用户再次访问该网站时,浏览器会将Cookie发送回服务器,服务器通过读取Cookie中的信息来识别用户身份或获取用户的个性化设置。

2.获取cookie的代码如下: 程序运行后会打开登录界面,然后使用的二维码登录。这样程序可以得到当前的cookie值字段,在cookie失效前无须再进行登录。

<0001>初始化浏览器的函数— browser_initial():

def browser_initial():
    """"
    进行浏览器初始化
    """
    # os.chdir('E:\\pythonwork')

    # 1.将加载项配置到启动浏览器中 打开/创建浏览器对象
    browser = webdriver.Chrome(service=Service(path))
    # 2.网址 https://kyfw.12306.cn/otn/resources/login.html -- 登录界面
    # https://kyfw.12306.cn/otn/leftTicket/init 选票界面可以 判断是否登录
    log_url = 'https://kyfw.12306.cn/otn/resources/login.html'
    return log_url, browser

<0002> 获得cookie的函数— get_cookies(browser):

def get_cookies(browser):
    """
    获取cookies保存至本地
    """
    browser.get(log_url)
    time.sleep(15)  # 进行扫码
    # selenium_login(browser)
    # print("登录成功")
    dictCookies = browser.get_cookies()  # 获取list的cookies
    string_Cookies = json.dumps(dictCookies)  # 转换成字符串保存
    with open('string_cookies.json', 'w') as f:
        f.write(string_Cookies)
    print('cookies保存成功!')

if __name__ == "__main__":
    log_url, browser = browser_initial()
    browser.get(log_url)
    # 1.二维码登录 保存 -- cookie -- 用完关掉
    get_cookies(browser)

运行后就会变成这样,找到二维码,让手机自动登录。
在这里插入图片描述
用手机扫描即可,这个操作是设置了时间time.sleep(15),你也可以在get_cookies(browser)中修改此代码,只有自己扫了登录成功,后续出现的cookie保存成功,才是最新的cookie值。

在这里插入图片描述

二、cookie登录函数与自动化 – 可以实现cookie值的匹配–实现自动登录 – 无须手机验证码

1.login_cookie函数代码

代码如下(示例):

def login_cookie(log_url, browser):
    """
    从本地读取cookies并刷新页面,成为已登录状态
    """
    # browser.get(log_url)
    with open('string_cookies.json', 'r', encoding='utf8') as f:
        listCookies = json.loads(f.read())

    # 往browser里添加cookies
    for cookie in listCookies:
        browser.add_cookie(cookie)
   
    time.sleep(1)
    # 如果是已登录的界面,免登录应该打开这个界面 https://kyfw.12306.cn/otn/view/index.html
    # https://www.12306.cn/index/ --12306官网首页
    browser.get(log_url)
    return browser

2.selenium_run(driver)函数代码 – 这里是可以到预定,然后抢票成功要10分钟之内自己付款。

def selenium_run(driver):

    # 11.1 选择出发的城市--点击那个框
    driver.find_element(by="css selector", value="#fromStationText").click()
    # 11.2 选择出发的城市--选择城市
    driver.find_element(by="css selector", value="#fromStationText").send_keys(login["from_city"])
    # 11.3 选择出发的城市--回车确定
    driver.find_element(by="css selector", value="#fromStationText").send_keys(Keys.ENTER)

    # 12.1 选择目的的城市--点击那个框
    driver.find_element(by="css selector", value="#toStationText").click()
    # 12.2 选择目的的城市--选择城市
    driver.find_element(by="css selector", value="#toStationText").send_keys(login['to_city'])
    # 12.3 选择目的的城市--回车确定
    driver.find_element(by="css selector", value="#toStationText").send_keys(Keys.ENTER)

    # 13.1 选择出发的日期--点击那个框
    driver.find_element(by="css selector", value="#train_date").clear()
    # 12.2 选择出发的日期--选择城市
    driver.find_element(by="css selector", value="#train_date").send_keys(login['train_date'])
    # 12.3 选择出发的日期--回车确定
    driver.find_element(by="css selector", value="#train_date").send_keys(Keys.ENTER)
    # # 12.4 点击--显示全部可预订的车次
    # driver.find_element(by="css selector", value="avail_ticket").click()
    # 12.5 点击查询
    driver.implicitly_wait(5)
    driver.find_element(by="css selector", value="#query_ticket").click()
    # num = int(input('请输入您想要的车次:'))
    # 点击预订按钮 -- 选择要预定的是第几躺车 -- 这个代码没有问题
    # driver.find_element(by="css selector", value='#ticket_65000G279007_01_03 > td.no-br').click()

    # //div//tbody[@id="queryLeftTable"]/tr[contains(@id,"ticket")]/@id -- 但是selenium的find_elements不可以用@id找到元素。
    ticket_list = driver.find_elements(by="xpath", value='//div//tbody[@id="queryLeftTable"]/tr[contains(@id,"ticket")]')
    ticket_num = ticket_list[login['num']].get_attribute("id")
    ticket_pre = ticket_num + ' > td.no-br'
    # 12.6 点击想要车次的预定按钮
    driver.find_element(by="css selector", value=f"#{ticket_pre}").click()

    # 13.1 勾选想要的乘车人
    driver.find_element(by="css selector", value='#normalPassenger_0').click()

    # 13.2 提交订单
    driver.find_element(by="css selector", value='#submitOrder_id').click()
    driver.implicitly_wait(5)
    # 13.3 选择靠窗的1F的位置 #erdeng1 > ul:nth-child(4) > li:nth-child(2) 是在浏览器右键copy selector选择出来的
    # # 1F是不可以选中的
    driver.find_element(by="css selector", value='#erdeng1 > ul:nth-child(4) > li:nth-child(2)').click()
    # 13.4 再次确认提交
    time.sleep(1)
    # 如果此处没有响应多点击就好 -- 加上 "try -- except"
    driver.find_element(by="css selector", value='#qr_submit_id').click()
    # driver.find_element(by="css selector", value='#qr_submit_id').click()
    # driver.find_element(by="css selector", value='#qr_submit_id').click()

三、完整的代码

1.代码主体 – 先运行get_cookies(browser),注释掉后面的两句代码,二维码登录以后,再注释掉get_cookies(browser),运行后面两个即可。

if __name__ == "__main__":
    log_url, browser = browser_initial()
    browser.get(log_url)
    # 1.二维码登录 保存 -- cookie -- 用完关掉
    get_cookies(browser)
    # 2.自动登录
    driver = login_cookie(log_url, browser)
    selenium_run(driver)


2.完整代码

import time
import json
from selenium import webdriver
from selenium.webdriver.chrome.service import Service
from selenium.webdriver.common.keys import Keys
path = r'D:\game\chromedriver.exe'


def browser_initial():
    """"
    进行浏览器初始化
    """
    # os.chdir('E:\\pythonwork')

    # 1.将加载项配置到启动浏览器中 打开/创建浏览器对象
    browser = webdriver.Chrome(service=Service(path))
    # 2.网址 https://kyfw.12306.cn/otn/resources/login.html -- 登录界面
    # https://kyfw.12306.cn/otn/leftTicket/init 选票界面可以 判断是否登录
    log_url = 'https://kyfw.12306.cn/otn/leftTicket/init'
    return log_url, browser


def get_cookies(browser):
    """
    获取cookies保存至本地
    """
    browser.get(log_url)
    time.sleep(15)  # 进行扫码
    # selenium_login(browser)
    # print("登录成功")
    dictCookies = browser.get_cookies()  # 获取list的cookies
    string_Cookies = json.dumps(dictCookies)  # 转换成字符串保存
    with open('string_cookies.json', 'w') as f:
        f.write(string_Cookies)
    print('cookies保存成功!')

def login_cookie(log_url, browser):
    """
    从本地读取cookies并刷新页面,成为已登录状态
    """
    # 这个是12306的登录界面 https://kyfw.12306.cn/otn/resources/login.html
    # browser.get(log_url)
    with open('string_cookies.json', 'r', encoding='utf8') as f:
        listCookies = json.loads(f.read())

    # 往browser里添加cookies
    for cookie in listCookies:
        # cookie_dict = {
        #     'domain': cookie.get('domain'),
        #     # "expires": cookie.get('value'),
        #     'httpOnly': cookie.get('httpOnly'),
        #     'name': cookie.get('name'),
        #     'path': cookie.get('path'),
        #     "sameSite": "Lax",
        #     'Secure': cookie.get('Secure'),
        #     'value': cookie.get('value'),
        # }
        browser.add_cookie(cookie)
    # browser.refresh()
    time.sleep(1)
    # 如果是已登录的界面,可以免登录直接进 https://kyfw.12306.cn/otn/view/index.html
    # https://www.12306.cn/index/ --12306官网首页
    browser.get(log_url)
    return browser

def selenium_run(driver):

    # 11.1 选择出发的城市--点击那个框
    driver.find_element(by="css selector", value="#fromStationText").click()
    # 11.2 选择出发的城市--选择城市
    driver.find_element(by="css selector", value="#fromStationText").send_keys(login["from_city"])
    # 11.3 选择出发的城市--回车确定
    driver.find_element(by="css selector", value="#fromStationText").send_keys(Keys.ENTER)

    # 12.1 选择目的的城市--点击那个框
    driver.find_element(by="css selector", value="#toStationText").click()
    # 12.2 选择目的的城市--选择城市
    driver.find_element(by="css selector", value="#toStationText").send_keys(login['to_city'])
    # 12.3 选择目的的城市--回车确定
    driver.find_element(by="css selector", value="#toStationText").send_keys(Keys.ENTER)

    # 13.1 选择出发的日期--点击那个框
    driver.find_element(by="css selector", value="#train_date").clear()
    # 12.2 选择出发的日期--选择城市
    driver.find_element(by="css selector", value="#train_date").send_keys(login['train_date'])
    # 12.3 选择出发的日期--回车确定
    driver.find_element(by="css selector", value="#train_date").send_keys(Keys.ENTER)
    # # 12.4 点击--显示全部可预订的车次
    # driver.find_element(by="css selector", value="avail_ticket").click()
    # 12.5 点击查询
    driver.implicitly_wait(5)
    driver.find_element(by="css selector", value="#query_ticket").click()
    # num = int(input('请输入您想要的车次:'))
    # 点击预订按钮 -- 选择要预定的是第几躺车 -- 这个代码没有问题
    # driver.find_element(by="css selector", value='#ticket_65000G279007_01_03 > td.no-br').click()

    # //div//tbody[@id="queryLeftTable"]/tr[contains(@id,"ticket")]/@id -- 但是selenium的find_elements不可以用@id找到元素。
    ticket_list = driver.find_elements(by="xpath", value='//div//tbody[@id="queryLeftTable"]/tr[contains(@id,"ticket")]')
    ticket_num = ticket_list[login['num']].get_attribute("id")
    ticket_pre = ticket_num + ' > td.no-br'
    # 12.6 点击想要车次的预定按钮
    driver.find_element(by="css selector", value=f"#{ticket_pre}").click()

    # 13.1 勾选想要的乘车人
    driver.find_element(by="css selector", value='#normalPassenger_0').click()

    # 13.2 提交订单
    driver.find_element(by="css selector", value='#submitOrder_id').click()
    driver.implicitly_wait(5)
    # 13.3 选择靠窗的1F的位置 #erdeng1 > ul:nth-child(4) > li:nth-child(2) 是在浏览器右键copy selector选择出来的
    # # 1F是不可以选中的
    driver.find_element(by="css selector", value='#erdeng1 > ul:nth-child(4) > li:nth-child(2)').click()
    # 13.4 再次确认提交
    time.sleep(1)
    # 如果此处没有响应多点击就好 -- 加上 "try -- except"
    # driver.find_element(by="css selector", value='#qr_submit_id').click()
    # driver.find_element(by="css selector", value='#qr_submit_id').click()
    # driver.find_element(by="css selector", value='#qr_submit_id').click()



f = open('login.json', encoding='utf-8').read()
login = json.loads(f)

if __name__ == "__main__":
    log_url, browser = browser_initial()
    browser.get(log_url)
    # 1.二维码登录 保存 -- cookie -- 用完关掉
    # get_cookies(browser)
    # 计算程序运行时间
    # start = time.perf_counter()
    # 2.自动登录

    driver = login_cookie(log_url, browser)
    # # 如果没有登录那就自动登录 -- 然后获得cookie自动保存
    # # if browser.find_element(by="css selector", value="#login_user").text == "登录":
    # # 3.自动买票
    selenium_run(driver)
    # end = time.perf_counter()
    # runTime = end - start
    # runTime_ms = runTime * 1000
    # # 输出运行时间
    # print("运行时间:", runTime, "秒")
    # print("运行时间:", runTime_ms, "毫秒")

3.login.json – 解释这个文件

在这里插入图片描述

图中字典信息:代表的是地址日期,num代表的是第几躺车,id_card身份证,后4位。这里面有用账户密码登录的信息,虽然这里用不上,但是我们上一个帖子是用这个的,所以我把它写在了一起。

四、总结

1.几个注意:代码中的url最好是用登录成功的页面,而非登录页面。不然会一直显示要登录。


log_url = 'https://kyfw.12306.cn/otn/resources/login.html'
log_url = 'https://kyfw.12306.cn/otn/leftTicket/init'  ##---用这个

2.改进:今天真正的实战我也挺满意的就是想着再改进一下,看看多线程是否可以直接抢到。明天就要真正抢票啦 – 给大家看看今天的战果。

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/950545.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

DNS协议漏洞利用实验_hust计算机网络安全实验

文章目录 计算机网络安全实验 DNS协议漏洞利用实验 docker使用 建立实验环境docker常用指令 一些注意事项设置本地 DNS 服务器 配置用户计算机设置本地DNS服务器在本地 DNS 服务器中建一个区域 修改主机文件&#xff08;可略&#xff09;netwox实施DNS的用户响应欺骗攻击netwo…

基于MP157AAA的I2C练习

练习要求&#xff1a; 通过I2C分别实现与芯片si7006(获取湿度、温度)和芯片ap3216(获取环境光照强度)的通讯&#xff1b; 1、运行效果 2、分析ap3216如何获取光照强度 2.1、需要操作的寄存器 通过分析手册&#xff0c;需要操作以下寄存器: 0x00&#xff1a;系统配置 0x0C&…

【Linux】深入理解文件系统(超详细)

目录 一.磁盘 1-1 磁盘、服务器、机柜、机房 &#x1f4cc;补充&#xff1a; &#x1f4cc;通常网络中用高低电平&#xff0c;磁盘中用磁化方向来表示。以下是具体说明&#xff1a; &#x1f4cc;如果有一块磁盘要进行销毁该怎么办&#xff1f; 1-2 磁盘存储结构 ​编辑…

网络安全图谱以及溯源算法

​ 本文提出了一种网络攻击溯源框架&#xff0c;以及一种网络安全知识图谱&#xff0c;该图由六个部分组成&#xff0c;G <H&#xff0c;V&#xff0c;A&#xff0c;E&#xff0c;L&#xff0c;S&#xff0c;R>。 1|11.知识图 ​ 网络知识图由六个部分组成&#xff0c…

《Spring Framework实战》7:4.1.2.容器概述

欢迎观看《Spring Framework实战》视频教程 容器概述 该接口表示 Spring IoC 容器&#xff0c;并负责实例化、配置和组装 bean。 容器在组件上获取其指令&#xff0c;以实例化、配置和 通过读取配置元数据进行汇编。可以表示配置元数据 作为带注释的组件类、具有工厂方法的配置…

学生公寓技术规格书如何编写?

学生公寓限电柜的技术规格书主要包括以下内容‌&#xff1a; ‌用电计量计费‌&#xff1a;限电柜可以通过计算机售电管理系统进行用电计量计费&#xff0c;学生需要预交电费&#xff0c;系统会自动将数据传给控电柜和配电箱&#xff0c;对宿舍的电量进行累减计量‌。 ‌时间控…

【HarmonyOS NEXT】鸿蒙应用点9图的处理(draw9patch)

【HarmonyOS NEXT】鸿蒙应用点9图的处理&#xff08;draw9patch&#xff09; 一、前言&#xff1a; 首先在鸿蒙中是不支持安卓 .9图的图片直接使用。只有类似拉伸的处理方案&#xff0c;鸿蒙提供的Image组件有与点九图相同功能的API设置。 可以通过设置resizable属性来设置R…

SpringBoot 使用 Cache 集成 Redis做缓存保姆教程

1. 项目背景 Spring Cache是Spring框架提供的一个缓存抽象层&#xff0c;它简化了缓存的使用和管理。Spring Cache默认使用服务器内存&#xff0c;并无法控制缓存时长&#xff0c;查找缓存中的数据比较麻烦。 因此Spring Cache支持将缓存数据集成到各种缓存中间件中。本文已常…

MySQL事件功能简介

MySQL 的事件调度器&#xff08;Event Scheduler&#xff09;提供了一种便捷的方法来定时执行 SQL 语句&#xff0c;从而实现数据维护、报告生成等自动化操作。本文将详细介绍 MySQL 的事件功能&#xff0c;并说明如何使用 Navicat 管理这些事件。 1. 什么是 MySQL 事件调度器…

高光谱相机的特点

光谱特性 高光谱分辨率&#xff1a;能将光谱范围分割成极窄的波段&#xff0c;光谱分辨率通常达到纳米级甚至亚纳米级&#xff0c;可精确捕捉到不同物质在细微光谱差异上的特征&#xff0c;比如可以区分不同种类的植被因叶绿素含量等差异而在光谱上的细微变化。 多波段探测&a…

备考蓝桥杯:数据结构概念浅谈

目录 1数据结构的概念 什么是数据结构: 为什么要有数据结构 2.数据结构的三个组成要素 1.逻辑结构 2.存储结构 3.数据运算 3。算法好坏的度量&#xff08;时间复杂度和空间复杂度&#xff09; 时间复杂度计算 最优和平均和最差时间复杂度 计算时间复杂度例子 空间复…

闲谭SpringBoot--ShardingSphere分库分表探究

文章目录 1. 背景2. 创建数据库3. 修改yml配置文件4. 分片算法类5. 测试6 小结 1. 背景 接上文&#xff0c;我们对日志表&#xff0c;进行了按月的分表&#xff0c;这样每个月几百万条数据量还是扛得住的。 但是如果数据再多呢&#xff0c;除了提高硬件性能&#xff0c;还有一…

基于伪分布式模式部署Hadoop集群

1.上传Hadoop安装包 在/export/software目录下使用rz命令上传Hadoop安装包 2.创建目录 在/export/servers目录下创建wfb-hadoop目录&#xff0c;用于存放Hadoop的安装目录&#xff0c;命令如下&#xff1a; mkdir -p /export/servers/wfb-hadoop 3.安装Hadoop 1)将Hadoop安…

Android车载音频系统目录

目录 第一章 1.1 Android Automotive&#xff08;一&#xff09; 1.2 Android Automotive&#xff08;二&#xff09; 1.3 Android Automotive&#xff08;三&#xff09; 第二章 2.1 Android车载音频系统概览 2.2 车载音频焦点 2.3 车载音频配置 2.4 Audio control HAL…

怎么管理电脑usb接口,分享四种USB端口管理方法

怎么管理电脑usb接口&#xff0c;分享四种USB端口管理方法 USB接口作为电脑重要的外部接口&#xff0c;方便了数据传输和设备连接。 然而&#xff0c;不加管理的USB接口也可能带来安全隐患&#xff0c;例如数据泄露、病毒传播等。 因此&#xff0c;有效管理电脑USB接口至关重…

React+redux项目搭建流程

1.创建项目 create-react-app my-project --template typescript // 创建项目并使用typescript2.去除掉没用的文件夹&#xff0c;只保留部分有用的文件 3.项目配置&#xff1a; 配置项目的icon 配置项目的标题 配置项目的别名等&#xff08;craco.config.ts&…

conda+jupyter+pycharm:如何在Windows conda环境下运行jupyter并使用浏览器或者pycharm运行.ipynb

1 安装conda 2 conda环境下安装jupyter pip install jupyter3 设置jupyter配置文件 1&#xff09;创建 jupyter_notebook_config.py文件 jupyter notebook --generate-config 2&#xff09;设置密码 3&#xff09;设置参数 直接将以下参数修改为自己的配置后复制到配置文件…

微信小程序获取图片使用session(上篇)

概述&#xff1a; 我们开发微信小程序&#xff0c;从后台获取图片现实的时候&#xff0c;通常采用http get的方式&#xff0c;例如以下代码 <image class"user_logo" src"{{logoUrl}}"></image>变量logoUrl为ur图片l的请求地址 但是对于很多…

【江协STM32】9-1/2/3 USART串口协议、USART外设、串口发送串口发送+接收

1. 通信接口 通信的目的&#xff1a;将一个设备的数据传送到另一个设备&#xff0c;扩展硬件系统通信协议&#xff1a;制定通信的规则&#xff0c;通信双方按照协议规则进行数据收发全双工&#xff1a;指通信双方能够同时进行双向通信。发送线路和接收线路互不影响&#xff0c…

第一 二章 小车硬件介绍-(全网最详细)基于STM32智能小车-蓝牙遥控、避障、循迹、跟随、PID速度控制、视觉循迹、openmv与STM32通信、openmv图像处理、smt32f103c8t6

第一篇-STM32智能小车硬件介绍 后续章节也放这里 持续更新中&#xff0c;视频发布在小B站 里面。这边也会更新。 B站视频合集: STM32智能小车V3-STM32入门教程-openmv与STM32循迹小车-stm32f103c8t6-电赛 嵌入式学习 PID控制算法 编码器电机 跟随 小B站链接:https://www.bilib…