#写在前面,这个程序我已经弄出来了,但是因为黄牛泛滥以及懒人太多,整个程序的代码就不贴出来了,这里纯粹就是技术交流。

只做技术交流、、、、、

嗯,程序结束后,自己还是得手动付款。

废话不多说,下面就直接开始技术主要部分阐述。

先讲理论部分:首先我们需要代码实现一个浏览器功能,那么模块基本上可以确定urllib.parse、urllib.request,这两个包都是和网址有关的模块,那么咱们去登录一个网址,特别是有验证码这些的网址,我们登录进去是不是就行了?答案是对的,但是我们用代码实现的话,这个网址可能每次都有可能被代码去请求,那么服务器怎么知道我们是一个人,而不是多个浏览器不同的用户呢?

此时cookie就非常重要了,在代码中设置好cookie,那么对方服务器自然就知道我们是一个人,比较服务器都是这么区分的。python3中 cookie这个功能是封装在http.cookiejar这个模块之内。好了,代码如下:

# coding=utf-8
# author: Jason
# time:2018/1/16 20:00:00
#version:1.0
import urllib.request as ul
import urllib.parse as uz
import http.cookiejar as cookielib
from json import loads
c=cookielib.LWPCookieJar()#先把cookie对象存储为cookiejar的对象
cookie = ul.HTTPCookieProcessor(c)#把cookiejar对象转换为一个handle
opener = ul.build_opener(cookie)#建立一个模拟浏览器,需要handle作为参数
ul.install_opener(opener)#安装一个全局模拟浏览器,代表无论怎么访问都是一个浏览器操作而不是分开获取验证码等msg

好了,如此一来,我们代码的初步实现已经完成,接下来就是进入网络分析部分

首先可以使用google浏览器或者搜狗浏览器(本人用的搜狗),打开F12,也就是开发者模式,登录12306的登录地址https://kyfw.12306.cn/otn/login/init

两个红圈中第二个是验证码来源,此时我们只需要记录这个网页(点进去)的详细情况,写入代码当中,python3中urllib.request这个模块打开既可

如此便是验证码来源,那么如何用代码捕捉呢?首先我们可以先乱输入密码,乱点验证码,然后我们直接点击登录

多了一个很奇妙的东西,此时,这里就是验证码验证的网址,那么我们是不是应该记录下来呢?很简单,到Headers里面就全都看得到了

上面那个是服务器验证网址,下面就是我们回复给他的东西,那么那个163,121其实就是我乱点的验证码坐标了。至于为啥是坐标,因为它是用鼠标去点图片,那么他只可能是记录坐标,除非他自己搞了一套人工智能验证图片,但基于他几年前就这么玩了,人工智能根本没有怎么开始,他自然只能是最原始的技术而已。

那么这代表了他是先验证验证码,那么验证密码的在哪?自然是需要验证码这关能过,那我们输一个正确的验证码,再写个错密码,登录

此时可以看到,和验证码一样的方法,我们的回复与验证都在这几个圈了,还记得上面验证码失败的时候回复给我们的code是不是有个数字?这个也很重要,那么可以看看我们的验证成功的验证码返回给我们的是什么东西

这次我们看到了,验证码成功,显示是4,好,那我们不就可以进行条件判断了么?

那么如何打开一个网址然后把我们点的东西一起发过去呢?上代码

headers={'User-Agent':'Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/63.0.3239.84 Safari/537.36'}#先写个头,表示我这是浏览器用户登录而不是代码登录,如果不写,代码默认用的签名之类的是编程语言的标识,这样对方服务器很容易就发现你是个脚本了

def get_code():#获取验证码的步骤req = ul.Request('https://kyfw.12306.cn/passport/captcha/captcha-image?login_site=E&module=login&rand=sjrand&0.6758635422370105')req.headers = headerscode_file = opener.open(req).read()#此时为浏览器的open而不再是ul.urlopen,下同with open(r'C:\Users\Administrator\Desktop\12306自动抢票\code.png','wb')as f:f.write(code_file)

把验证码直接下载后方电脑上,后面要坐标只需要打开这个图既可输入,坐标的输入方式我用字典表示给大家看{1:(45,45)}{2:(120,45)}{3:(180,45)}{4:(255,45)}{5:(45,120)}{6:(120,120)}{7:(180,120)}{8:(255,120)}

根据这个验证码的排序,我相信读者应该知道顺序怎么来的吧,比较坐标就能懂了。

继续

def main_():get_code()code = input('输入验证码:')req = ul.Request('https://kyfw.12306.cn/passport/captcha/captcha-check')req.headers = headersdata = {'answer':code,'login_site':'E','rand':'sjrand'}data = uz.urlencode(data).encode()#把字典转换为URL query string,此时是str,要把它变为byts。
html = opener.open(req,data= data).read().decode()#读取出来是byts格式,转换为‘utf-8(默认)print(html)result = loads(html)if result['result_code']=='4':print('验证码通过')rep = ul.Request('https://kyfw.12306.cn/passport/web/login')rep.headers = headersdata = {'username':'这里就是你用户名','password':'这里就是你的密码','appid':'otn'}data = uz.urlencode(data).encode() #看到了吗,这就是你给服务器回复的东西html1 = opener.open(rep,data = data ).read().decode()result1 = loads(html1)if result1['result_code'] == 0:print('账户密码验证通过')else:print(result1['result_message'])else:print('验证码校验失败,重来')if __name__ == '__main__':main_()

此时,咱们就过了验证码密码这一关,后面是不是又要查票?那么同样的方法,我们就可以以此类推到最后一步,这里就不一一贴代码了

ps:查代码这几步的信息可是很重要喔,我们要把它记录好,并且这里面的信息包含了各种作为信息以及他们的顺序,多测试几次基本都能搞出来,这里就是提醒一点

找找规律,然后用split的方法完全就可以切割出来,然后一个循环,就可以得到第几个元素是我们要的,那么后面就可以标志判断返回值如果是无,就没票可以继续查询,直到有票就可以下一步;

那么有票的话,后面一样也是以此类推的方式,代码我就不重现了,很简单,我就把后面出现需要请求的网址都发出来供大家观摩

查询车票信息

url = 'https://kyfw.12306.cn/otn/leftTicket/queryZ?leftTicketDTO.train_date=%s&leftTicketDTO.from_station=%s&leftTicketDTO.to_station=%s&purpose_codes=ADULT'%(train_data,from_station,to_station)

req = ul.Request('https://kyfw.12306.cn/otn/leftTicket/submitOrderRequest')#确定订单信息
req = ul.Request("https://kyfw.12306.cn/otn/confirmPassenger/initDc")#验证订单
req = ul.Request('https://kyfw.12306.cn/otn/confirmPassenger/getPassengerDTOs')#准备跨到下单中的过度
req = ul.Request('https://kyfw.12306.cn/otn/confirmPassenger/checkOrderInfo')#检查订单信息
req = ul.Request('https://kyfw.12306.cn/otn/confirmPassenger/getQueueCount')#信息提交给服务器,准备进入下单步骤
req = ul.Request('https://kyfw.12306.cn/otn/confirmPassenger/confirmSingleForQueue')#正式进入下单步骤
req = ul.Request('https://kyfw.12306.cn/otn/confirmPassenger/queryOrderWaitTime?random=%s&tourFlag=dc&_json_att=&REPEAT_SUBMIT_TOKEN=%s'%(numb,time.time()))#下单确认中,此时这个网址一般是进行两次访问,不知为何,我还是做了两次访问,numb是前面查询车票点击预定回复我们的信息中的一条,有点难找喔,我曾经找了三天。。。当然是因为自己不仔细而已
zreq = ul.Request("https://kyfw.12306.cn/otn/confirmPassenger/resultOrderForDcQueue")#最后的结果回执,如果一切都顺利,那么票就已经订了。我一般是打印他返回的内容
'''zreq = ul.Request("https://kyfw.12306.cn/otn/confirmPassenger/resultOrderForDcQueue")zreq.headers = headersdata ={"REPEAT_SUBMIT_TOKEN":"%s"%numb,"_json_att": "","orderSequence_no":orderId}data = uz.urlencode(data).encode()html = opener.open(zreq,data=data).read().decode()result = loads(html)print('代码全部过完,回去登录下是否搞定')print(result)print(result['data']['submitStatus'])if result['data']['submitStatus'] == True:print('购票成功')return Trueelse:print('购票失败,重试其他列车')continue
'''
最终的回执代码详细 信息,读者可以自己尝试多次,得到自己的回复代码确认是否购票成功,因为result['data']['submitStatus']==True只不过是确认订单状态而已,这个被我改动过,你可以多次尝试

最后的最后,火车票预订成功只有30分钟支付时间,所以我为了防止订好票但是我人不在,特意写了qq邮件通知

qq邮件通知:

def email():#这是我订票后给自己发邮件的函数import smtplibfrom email.mime.text import MIMETextimport timetext = '已经为%s抢到票,速度登录12306付款,用户名:%s,密码:%s'%(NAME,username,password)msg = MIMEText(text, 'plain', 'utf-8')msg_From = '2059****16@qq.com'msg_To = '5043****75@qq.com'#是的,我有两个qq,一个发一个收
smtpSever = 'smtp.qq.com'  # qq邮箱的smtp Sever地址smtpPort = '465'  # 开放的端口sqm = 'q********eghe'  # 在登录smtp时需要login中的密码应当使用授权码而非账户密码
msg['from'] = msg_Frommsg['to'] = msg_Tomsg['subject'] = 'Python自动邮件-%s' % time.ctime()smtp = smtplibsmtp = smtplib.SMTP_SSL()'''smtplib的connect(连接到邮件服务器)、login(登陆验证)、sendmail(发送邮件)'''smtp.connect(smtpSever, smtpPort)smtp.login(msg_From, sqm)smtp.sendmail(msg_From, msg_To, str(msg))# s = smtplib.SMTP("localhost")# s.send_message(msg)
    smtp.quit()print('邮件已发送~你可以安心去玩了')def emailforcode():#此函数是防止查询有票但12306账号掉线人不在无法订票的提醒import smtplibfrom email.mime.text import MIMETextimport timetext = '%s账号下线,速度登录验证12306' % NAMEmsg = MIMEText(text, 'plain', 'utf-8')msg_From = '205****516@qq.com'msg_To = '50****75@qq.com'smtpSever = 'smtp.qq.com'  # qq邮箱的smtp Sever地址smtpPort = '465'  # 开放的端口sqm = 'qowa*******ghe'  # 在登录smtp时需要login中的密码应当使用授权码而非账户密码
msg['from'] = msg_Frommsg['to'] = msg_Tomsg['subject'] = 'Python自动邮件-%s' % time.ctime()smtp = smtplibsmtp = smtplib.SMTP_SSL()'''smtplib的connect(连接到邮件服务器)、login(登陆验证)、sendmail(发送邮件)'''smtp.connect(smtpSever, smtpPort)smtp.login(msg_From, sqm)smtp.sendmail(msg_From, msg_To, str(msg))# s = smtplib.SMTP("localhost")# s.send_message(msg)
    smtp.quit()print('邮件已发送~')

如此就大功告成了。

不能发完整的代码(本身目的就是为了技术交流而已,防止懒人盗码乱搞),但是我相信各位开发中的朋友们只要有逻辑,有开头,只要自己肯动手,都可以自己钻研出来,举一反三。毕竟我就是这样搞出来的,我从来都相信,只要肯学,都会学会,只要肯做,都可以做成。

好了,人生苦短,我用python,我是一枚土木人转行搞开发的小白。一切全靠自学,一起加油,共勉~

转载于:https://www.cnblogs.com/Jason504327775/p/8525189.html

python实现12306抢票以及自动邮件发送提醒付款相关推荐

  1. 快过年了,Python实现12306查票以及自动购票....

    嗨害大家好鸭!我是小熊猫~ 明天就是2023年啦~ 还有谁像我小熊猫一样没有回家的? 这次康康能不能12306抢票回家!!! Python实现12306查票以及自动购票 [代码来源]: 青灯教育-自游 ...

  2. 通过python实现12306抢票

    铁路12306抢票系统 ​ #!/usr/bin/env python # -*- coding: utf-8 -*- """ 通过splinter刷12306火车票 进 ...

  3. 用python实现12306抢票

    用splinter或者selenium都可以实现.这两个库是实现web浏览器自动操作的库.就是模拟人的点击等等一系列操作. 不过先对应你的谷歌浏览器版本下好对应的chromedriver,然后添加好环 ...

  4. Python实现12306查票以及自动抢票

    市场上很多火车票抢票软件大家应该非常熟悉,但很少有人研究具体是怎么实现的,所以觉得很神秘,其实很简单.下面使用Python模拟抢票程序,给大家揭秘抢票到底是怎么回事. 目录 环境使用 相关模块 思路 ...

  5. 12306抢票脚本 python_如何使用Python实现12306抢票?摆脱无票可买的窘境

    前言 十一已经过去一个星期了,下一个假期就是元旦啦,每一次假期购票都得抢到"头破血流",所以小编经历过这次十一之后就在想做一个抢票小助手,经过几天的构思后,终于写了出来. 一.爬虫 ...

  6. python开发12306抢票_如何利用 Python 实现12306抢票?

    生苦短,我用 python. 作为一种"胶水语言",python 为无数码农带来了便利:同时,越来越多的 python 工程师被标榜为"高薪党":"全 ...

  7. Python + Splinter 12306抢票

    临近春节和期末放假,很多小伙伴也开始关注起来了12306的放票与抢票,学习python之余,敲一个小的demo,如果能帮到人成功抢到票,那便是最好不过的了.这个小的脚本完成过程中,做了很多的调试,力求 ...

  8. Python操作12306抢票脚本

    有一段时间没有使用Python了,前几天经朋友提起一篇关于用Python实现抢火车票的文章,百度了实现抢火车票的技术细节,网上却有不少资料,也不是新鲜的东西.在了解了一些技术手段,阅读了一些大神的博文 ...

  9. python抢票web端_python+Splinter实现12306抢票功能

    本文实例为大家分享了python实现12306抢票功能的具体代码,供大家参考,具体内容如下 源码记录如下: #!/usr/bin/env python # _*_ coding:utf-8 _*_ # ...

最新文章

  1. spark-submit参数说明--on YARN
  2. 基于工程经验的『RESTful接口设计规范』
  3. Swoole的think-swoole的安装
  4. Vue学习笔记进阶篇——Render函数
  5. [shell进阶]——shell多线程
  6. maven exclude java_java – Maven:从shade插件中排除依赖项
  7. 课程题目 : 1003. 简单字符串匹配
  8. Linkerd2安装和使用
  9. pytorch int64的tensor怎么转换成float64
  10. python面向对象设计角色攻击_Python技能:面向对象基础实战之英雄联盟
  11. error while loading shared libraries: libtinfo.so.5
  12. boost::unorder_map如何插入元素_「leetcode」701. 二叉搜索树中的插入操作:【递归法】【迭代法】详解...
  13. 手艺人舍bpftrace而取systemtap的代价和思考
  14. win10修改计算机物理地址,Windows10系统修改物理地址的方法
  15. Android人脸支付功能,OPPO Find X成为首款支持微信人脸支付的安卓手机
  16. select添加提示语
  17. eval()与train()(结合源码理解)
  18. javaweb-39:文件上传及拓展鸡汤
  19. 我的世界Faithful Java_我的世界:原来我们都被骗了,这才Minecraft真实的样貌
  20. 结构体字节对齐详解【含实例】

热门文章

  1. [办公软件学习教程] 如何使用Excel高亮查找出来的单元格
  2. 基础研究不直接影响日常生活,但,它对人类文明至关重要!
  3. 《Android App开发入门与项目实战》出版后记
  4. 计算机看到硬盘是空白状态,我的可移动硬盘打不开了,插入后电脑上有显示,就是打不开,查看属性是满盘了,在电脑磁盘运行里看是空的...
  5. WPS和Office的自动备份目录
  6. android 停止 服务 命令,单击按钮启动和停止HCE HostApduService-Android[复制]
  7. access自动编号怎么解除_Access字段中“自动编号”类型不能再改回来的解决方法...
  8. 原生js获取和设置页面垂直滚动高度
  9. root全攻略(root是什么 怎么root root能干什么)
  10. 我赢助手专注于自媒体短视频领域四年后的运营问题汇总-第四集