用httpclient开发的在线自动抢订火车票系统(已上传源代码)

play 专栏收录该内容
9 篇文章 0 订阅
 前两天女友要在线订火车在票,一直都没有办法订到票,最后没办法,便给她写了个自动抢票的脚本,可周边的朋友听她说通过软件订到票了,都先后向她要软件,可原来的脚本只是基于控制台输入,这样给别人也没法用,兴趣一起,就花了一天的时间做了个WEB界面,然后分享给朋友用。先上几张图,看大家看看。

这个是登录界面,要使用前先设置一下常用的邮箱和登录密码,自动抢票过程中,如需要再输入登录验证码和订单验证码,或订单成功时,将通过此email通知您,建议使用qq邮箱,这样只要您在电脑时开打QQ,收到邮件时,qq会弹出窗口通知您,您可以即时响应。  
第一次使用时,要先把在火车票官网注册的资料填在这里:

这资料都必码是已在官方上注册过的,在自动订票时,需要用到这些资料。



第一次登录后需要求先填写官网上的登录验证码和订单提交验证码,只要填写正确后,在填票过程中就一直可以使用,这也是官网的BUG,应该是他们在较正验证码后,没有让当前验证码失效,这样就只要不去刷新验证码,就可以一直使用第一次输入的正确验证码。


填写一下订票任务,主要就填写订票人资料(当然这个也是要在官网上添加过的),和订票信息,什么时候从哪到哪,从几列车等,保存完后,点开始,就则可以自动登录,找票,然后订票,订票成功后发邮件通知。


执行过程中,会时间更新执行的结果信息。




留言板功能,收集问题的反馈。


网友提供http代理,官方网站对一个IP的单位时间内访问次数据有限制,超过这个超制值IP会被封锁一段时间。
因为使用的人多,所以系统需要使用代理来向官网发送请求。

看到系统的整体内容后,再来介绍一下相关的技术
这个自动发送请求,提交数据,都依赖于httpclient4,火车票订票官方是使用ssl加密,这里是启用httpclient的ssl功能
Java代码 复制代码  收藏代码
  1. SSLContext sslcontext = SSLContext.getInstance("TLS");   
  2. sslcontext.init(nullnew TrustManager[]{easyTrustManager}, null);   
  3. SSLSocketFactory sf = new SSLSocketFactory(sslcontext);   
  4. Scheme sch = new Scheme("https", sf,443);   
  5. httpclient.getConnectionManager().getSchemeRegistry().register(sch);  
大家都知道,这个官网的反映速度有时实在是惨不忍睹,所以我们也要设计一下请求等待的最长时间(30秒):
Java代码 复制代码  收藏代码
  1. httpclient.getParams().setIntParameter("http.socket.timeout",30000);//毫秒  
官方网站对一个IP的单位时间内访问次数据有限制,超过这个超制值IP会被封锁一段时间。
因为使用的人多,所以系统需要使用代理来向官网发送请求,这个代理需要支持ssl,所以先网上找到一批代理IP过来,存在一个txt文字,格式为 ip:端口\n,以下读取IP代理,并且进行速试测试
Java代码 复制代码  收藏代码
  1. public void run(){   
  2.       try {   
  3.         System.out.println("-----设置代理服务器----");   
  4.         String proxyFileString0 = FileUtils.readFileToString(new File(ApplicationUtils.getWebrootDir()+"/order/proxy_add.txt"), "UTF-8");   
  5.         String proxyFileString1 = FileUtils.readFileToString(new File(ApplicationUtils.getWebrootDir()+"/order/proxy_enable.txt"), "UTF-8");   
  6.         String proxyFileString = proxyFileString0+"\n"+proxyFileString1;   
  7.         for(String proxyString : StringUtils.split(proxyFileString, "\n")){   
  8.             proxyString = proxyString.trim();   
  9.             if(StringUtils.isNotEmpty(proxyString)){   
  10.                 System.out.print("-----测试代理服务器: "+proxyString);   
  11.                 String[] proxyInfo = proxyString.split(":");   
  12.                    
  13.                 HttpHost httpHost = new HttpHost(proxyInfo[0], Integer.valueOf(proxyInfo[1]));    
  14.                 long start = System.currentTimeMillis();   
  15.                 if(HttpUtils.testProxy(httpHost)){   
  16.                     long usetime = System.currentTimeMillis()-start;   
  17.                     if(usetime > 10000){   
  18.                         System.out.println(" 不使用,响应时间太长时间:"+usetime+"毫秒----");   
  19.                     }else{   
  20.                         System.out.println(" 可使用,使用时间:"+usetime+"毫秒----");   
  21.                         httpHostList.add(httpHost);   
  22.                     }   
  23.                 }   
  24.             }   
  25.         }   
  26.         System.out.println("-----设置代理服务器成功,总数:"+httpHostList.size()+"----");   
  27.     } catch (Exception e) {   
  28.     }   
  29. }  
最后在每次请求时,选择一个代理
Java代码 复制代码  收藏代码
  1. //设置代理对象 ip/代理名称,端口         
  2. try{   
  3.     httpclient.getParams().setParameter(ConnRoutePNames.DEFAULT_PROXY,ProxyHostUtils.next());   
  4. }catch(Exception e){}  
每个用户在发送请求前,先从官网上读取两个验证码,一个是登录验证码,一个是提交订单验证码,一次性读取就可以了。
Java代码 复制代码  收藏代码
  1. /**  
  2.      * 读取登录验证码  
  3.      * @throws Exception  
  4.      */  
  5.     @Transient  
  6.     public void getRemoteLoginCode() throws Exception {   
  7.         File file = HttpUtils.doGetFile(OrderRunTask.loginCodeUrl+"&nocache="+Utils.getRandomString(10), cookieContext);   
  8.         if(file != null){   
  9.             File codeFile = new File(ApplicationUtils.getWebrootDir()+getLoginCodeUrl());   
  10.             FileUtils.copyFile(file, codeFile);   
  11.         }   
  12.     }   
  13.        
  14.     /**  
  15.      * 读取提交订单验证码  
  16.      * @throws Exception  
  17.      */  
  18.     @Transient  
  19.     public void getRemoteSumitCode() throws Exception {   
  20.         File file = HttpUtils.doGetFile(OrderRunTask.submitCodeUrl+"&nocache="+Utils.getRandomString(10), cookieContext);   
  21.         if(file != null){   
  22.             File codeFile = new File(ApplicationUtils.getWebrootDir()+getSubmitCodeUrl());   
  23.             FileUtils.copyFile(file, codeFile);   
  24.         }   
  25.            
  26.     }  
每个用户可以同时订多张火车票,系统为每张火车票启动一个线程,持续的运行,直到订票成功或者是被中断。
Java代码 复制代码  收藏代码
  1. //建立线程池   
  2. public static ExecutorService runningTaskPool = Executors.newFixedThreadPool(1000);   
  3.   
  4. //启动订票任务   
  5. OrderRunTask orderRunTask = new OrderRunTask(account, orderTask);   
  6. OrderRunTaskUtils.runningTaskMap.put(orderTask.getId(), orderRunTask);   
  7. Future<?> future = OrderRunTaskUtils.runningTaskPool.submit(orderRunTask);   
  8. orderRunTask.setFuture(future);   
  9.   
  10. //中断任务   
  11. OrderRunTask orderRunTask = OrderRunTaskUtils.runningTaskMap.get(taskId);   
  12. try{   
  13.     orderRunTask.setStop();   
  14.     orderRunTask.getFuture().cancel(true);   
  15. }catch(Exception e){   
  16.     e.printStackTrace();   
  17. }   
  18.   
  19. //判断是否订票成功,并且发送邮件   
  20. if(body.indexOf("45分钟") != -1){   
  21.     Date now = new Date();   
  22.     account.sendMail(orderTask.getQueryTrainDate()+"_订票成功",   
  23.         orderTask.getQueryTrainDate() +" "+   
  24.         orderTask.getFromStation()+"到"+orderTask.getToStation()+(StringUtils.isEmpty(orderTask.getTrainNo())?orderTask.getTrainNo():"")+" --订票时间:"  
  25.         +DateUtils.getDateTime("HH:mm:ss", now)   
  26.     );   
  27.     account.setReload(true);   
  28.                                     orderTask.setState(200);   
  29.                                     orderTask.setRuningTime(now);   
  30.                                     this.account.error(orderTask.toString()+": 订票成功。");   
  31.     break next;   
  32. }  
这也只是本人兴趣开发,不使用在商品场合,以上分享的只是这个系统的设计思想,有时间再深入讨论。
顺便附上本机(随时变动的)的一个链接地址,地址已关闭,访问太多了,小小的笔记本,顶不住压力。
大伙这么好热情啊,现在已上传了控制台输入版本(jar包太大,没上传,需要的自己网上下载),欢迎大家讨论,至于web版本的,现在不好上传,里面有其他项目的内容,不好整理,这就不好意思了,见谅。

如果使用中报出javax.net.ssl.SSLException: hostname in certificate didn't match异常,是检查一下jar,httpclient4.0.1不会有这个问题,至少具体原因是没什么,还没有去调试。

如果报出java.lang.IllegalStateException: Invalid use of SingleClientConnManager: connection still allocated.
这个因为所有线程都共享一下httpclient,造成http发送请求的冲突,可以改造一下HttpUtils,为每个线程分配一个独立的httpclient,使用ThreadLocal; 可以更新一下HttpUtils。

下面有网友回复程序“开始搜索”后就停止了,因为程序对每个订票任务开始时只打印一下次"--开始搜索:"+ queryStrs[0]+"到"+queryStrs[1],在while中没有再打印搜索票过程的信息,所以看着是停止了,如果要看搜索过程只要在while里面打印消息提示就可以了。这在源代码中,可以很容易就发现的。


OrderMain中有句代码需要改一下,68行for(String train :StringUtils.split(queryBody, "\\n")){
之前这样是可以,但现在有人发现这个不行了,建议改成:
queryBody= queryBody.replace("\\n", "@@@");
for(String train :StringUtils.split(queryBody, "@@@")){
在开发时,这里也测试了用几遍,之前用queryBody.split(("\\n");是一直没有办法成功。不知网站返回来的字符串有问题,还是java的split有问题,看哪位能解释一下,OrderMain.rar已上传。


提交订单的时候,有两个参数设置出错,
Java代码 复制代码  收藏代码
  1. //设置座位信息   
  2. orderParam.put("passengerTickets", seatValue+",1,"+TrainDataUtils.name+",1,"+TrainDataUtils.cardid+","+TrainDataUtils.mobile+",Y"); orderParam.put("passenger_1_seat", seatValue);   
  3. //这个地方不需要   
  4.   
  5.   
  6. postOrderParam.putAll(TrainDataUtils.submitOrderParams);   
  7. //这里添加                                          postOrderParam.put("passengerTickets", seatValue+",1,"+TrainDataUtils.name+",1,"+TrainDataUtils.cardid+","+TrainDataUtils.mobile+",Y");                                         postOrderParam.put("passenger_1_seat", seatValue);  
OrderMain.rar已上传。


这份代码于1月9号之前可以成功订票,但不代表就一直可以订票,若官网程序一更新升级,肯定就不能订了,但分享源代码是为了与大家做技术交流,至于用不用于订票吧,自己决定,如果有出什么订票之类的问题,源代码都在,自己调试一下,一般是小改一下就可以订了(官方此时应该不会做太大的改动的),或者在从网上其他地方下载订票软件(目前挺多),所以这代码我不会去升级什么内容来让它可以一直保持能订票,但如果发现有什么设计上问题,会更新代码继续上传。
  • 大小: 64.8 KB
  • 大小: 75.2 KB
  • 大小: 48.7 KB
  • 大小: 64.4 KB
  • 大小: 25.3 KB
  • 大小: 20.7 KB
  • 大小: 22.5 KB
  • 0
    点赞
  • 1
    评论
  • 0
    收藏
  • 一键三连
    一键三连
  • 扫一扫,分享海报

©️2021 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值