全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

Java动态显示文件上传进度实现代码

本文实例实现文件上传的进度显示,我们先看看都有哪些问题我们要解决。

1 上传数据的处理进度跟踪
2 进度数据在用户页面的显示

就这么2个问题,
第一个问题,主要是组件的选择
必须支持数据处理侦听或通知的组件。当然,我肯定只用我自己的组件啦。基本原理是

1 使用request.getContentLength() 读取到处理数据的总长度,注意这个长度不等于文件的长度,因为Base64等编码会增加数据量,如果超过了允许的长度,直接返回-1;

2 在每读取一部分数据时(比如一行,或者64K,或者你自定义的字节数),将读取的字节数通知我们的进度跟踪程序。我取名为 UploadListener代码如下

/*
* 处理附件上传的通知。

* 各位可以继承这个类,来实现自己的特殊处理。 
* 
* @author 赵学庆 www.java2000.net 
*/ 
 public class UploadListener ... { 
 // 调试模式将在控制台打印出一些数据 
 private boolean debug;

 // 总数据字节数 
 private int total;

 // 当前已经处理的数据字节数 
 private int totalCurrent = 0 ;

 // 延迟,用来调试用,免得速度太快,根本卡看不到进度 
 private int delay = 0 ;

 /** */ /** 
 * 处理数据通知的方法。

 * 保存已经处理的数据。并且在一定的比例进行延迟。默认每1%

 * 如果不需用延迟,可以删掉内部的代码,加快速度。 
 * 
 * @param size 增加的字节数 
 */ 
 public void increaseTotalCurrent( long size) ... { 
 this .totalCurrent += size; 
 try ... { 
 currentRate = totalCurrent * 100 / total; 
  if (currentRate > lastRate) ... { 
  if (delay > 0 ) ... { 
  Thread.sleep(delay); 
  } 
  if (debug) ... { 
  System.out.println( " rate= " + totalCurrent + " / " + total + " / " + (totalCurrent * 100 / total)); 
  } 
  lastRate = currentRate; 
 } 
 } catch (Exception e) ... { 
 e.printStackTrace(); 
 } 
 } 

 /** */ /** 
 * 读取全部自己数 
 * 
 * @return 
 */ 
 public int getTotal() ... { 
 return total; 
 } 

 /** */ /** 
 * 读取已经处理的字节数 
 * 
 * @return 
 */ 
 public int getTotalCurrent() ... { 
 return totalCurrent; 
 } 

 private long lastRate = 0 ;

 private long currentRate = 0 ;

 public int getDelay() ... { 
 return delay; 
 } 

 public void setDelay( int delay) ... { 
 this .delay = delay; 
 } 

 public void setTotal( int total) ... { 
 this .total = total; 
 } 

 public boolean isDebug() ... { 
 return debug; 
 } 

 public void setDebug( boolean debug) ... { 
 this .debug = debug; 
 } 
}

3 下面我们来看上传的处理部分

 Upload upload = new Upload(request); 
 // 增加了侦听进度的代码 
 UploadListener uploadListener = new UploadListener(); 
 // 这句话我们后面再讨论,这个可是关键 
 session.setAttribute( " uploadListener " ,uploadListener); 
 uploadListener.setDelay( 0 ); 
 uploadListener.setDebug( true ); 
 upload.setUploadListener(uploadListener); 
 upload.parse(); 
 // 这句话同样重要,我们后面再讨论 
 session.setAttribute( " uploadListener " , null );

4 我们再看上传的表单部分

< script. type = " text/javascript. " > 
 function checkForm() ... { 
 $( " SHOW_FRAME. " ).src = " link.jsp " ; 
 $( ' SUBMIT ' ).disabled = true ; 
 Ext.MessageBox.show( ... { 
 title: ' Please wait... ' , 
 msg: ' Initializing... ' , 
 width: 240 , 
 progress: true , 
 closable: false 
 } ); 
 $( " MAIN_FORM. " ).submit(); 
 return false ; 
} 
 function setUploadProcess(total,current) ... { 
 var rate = Number(current) / Number(total); 
 Ext.MessageBox.updateProgress(rate, ' Uploading... ' + current + " / " + total); 
 if (Number(current) >= Number(total)) ... { 
 closeUploadProcess(); 
 } 
} 
 function closeUploadProcess() ... { 
 Ext.MessageBox.hide(); 
} 
</ script. > 
< iframe. name = " ACTION_FRAME. " id = " ACTION_FRAME. " width = " 0 " height = " 0 " ></ iframe. > 
< iframe. name = " SHOW_FRAME. " id = " SHOW_FRAME. " width = " 0 " height = " 0 " ></ iframe. > 
< form. method = " OST " id = " MAIN_FORM. " nsubmit = " return checkForm() " enctype = " multipart/form-data " 
 action = " uploadFileSave.jsp " target = " ACTION_FRAME. " > 
 < input type = " file " size = " 50 " name = " file " > 
 < input type = " submit " ID = " SUBMIT " value = " Upload It " > 
</ form. >

第一个iframe用于提交表单数据,第二个就是我们用来获取处理数据进度信息的。
提交表单很简单,target指向了我们的第一个iframe.
我们看一下JS
checkForm. 里面第一句就是关键的读取进度信息的页面,我们在第二个iframe里面获得。然后就是弹出进度的显示框,我使用了Ext. 然后提交上传表单
setUploadProcess 用来更新进度框上面的数据,第一个参数是数据总共的大小,第二个参数是已经处理的大小。
closeUploadProcess 关闭进度框

5 最后,我们来看读取进度信息的页面

<% @ page language = " java " contentType = " text/html; charset=utf-8 " pageEncoding = " utf-8 " %> 
<% @include file = " ../package.inc.jsp " %> 
<% 
 response.setHeader( " ragma " , " no-cache " ); 
 response.setHeader( " Cache-Control " , " no-cache " ); 
 response.setDateHeader( " Expires " , 0 ); 
 response.setBufferSize( 0 ); 
 UploadListener uploadListener = null ; 
 while (uploadListener == null || uploadListener.getTotalCurrent() <= 0 ) ... { 
 uploadListener = (UploadListener) session.getAttribute( " uploadListener " ); 
 out.print( " . " ); 
 out.flush(); 
 Thread.sleep( 10 ); 
 } 
 long total = uploadListener.getTotal(); 
 out.println(total); 
 long current; 
 out.flush(); 
 while ( true ) ... { 
 current = uploadListener.getTotalCurrent(); 
 if (current >= total) ... { 
 break ; 
 } 
 out.println( " <script. type='text/javascript'>parent.setUploadProcess(' " + total + " ',' " + current + " ');</script> " ); 
 out.flush(); 
 Thread.sleep( 10 ); 
 } 
%>< script. type = " text/javascript. " > parent.closeUploadProcess(); </ script. >

其中前面的循环,用来判断是否产生了上传的信息,如果没有则等待。

然后就是读取上传的信息,并计算后生成调用上级窗口的更新进度条的JS, 请注意out.print后面必须跟上out.flush,否则不会持续输出到客户端,也就不会看到连续的进度条变化。

总结:

上面的部分比较乱,我这里总结一下关键点。

1、在上传组件里面,把总大小和当前读取了的大小放到一个类里面,并持续更新,直到处理完毕
2、上传的进度类,放在session里面,供进度读取页面读取
3、进度读取页面,从session里面拿到数据,并返回结果。

有几个疑问解释一下。

1、由于Http协议决定了,必须等request处理完毕才会返回输出,所以不能在upload页面里进行处理进度的显示。我前面测试到1M左右的文件不成功,就是没有考虑到这个问题。所以必须单独用一个GET的程序进行读取
2、读取是一个持续不断的过程,因为上传大文件是很慢的!
3、如果你的应用服务器启用了GZIP压缩,是容器管理的,那么很不幸,因为容易必须拿到所有的数据,至少是一部分数据才会返回,所以造成我们返回的那些很少的字节经常会被截住,造成无法显示上传的连续过程。  

解决方法

1) 关闭GZIP, 我想许多人不会这么做 
2) 使用自定义的GZIP压缩,判断某些东西(比如URL),对他们不进行压缩处理

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# java动态显示文件上传进度  # java文件上传进度  # java文件上传  # java 教你如何给你的头像添加一个好看的国旗  # Java Struts图片上传至指定文件夹并显示图片功能  # Java Servlet上传图片到指定文件夹并显示图片  # JavaWeb项目实现文件上传动态显示进度实例  # Java web实现头像上传以及读取显示  # 上传  # 第一个  # 表单  # 第二个  # 自己的  # 才会  # 这句话  # 自定义  # 是一个  # 进度条  # 我想  # 放在  # 一句  # 也就  # 将在  # 这个问题  # 能在  # 很简单  # 如果没有  # 弹出 


相关文章: 建站之星价格显示格式升级,你的预算足够吗?  如何选择CMS系统实现快速建站与SEO优化?  建站之星代理商如何保障技术支持与售后服务?  Swift中switch语句区间和元组模式匹配  建站之星如何配置系统实现高效建站?  建站主机CVM配置优化、SEO策略与性能提升指南  Python路径拼接规范_跨平台处理说明【指导】  如何获取免费开源的自助建站系统源码?  IOS倒计时设置UIButton标题title的抖动问题  建站之星收费标准详解:套餐费用及年费价格表一览  官网网站制作腾讯审核要多久,联想路由器newifi官网  如何在IIS中新建站点并配置端口与物理路径?  网站制作企业,网站的banner和导航栏是指什么?  深圳企业网站制作设计,在深圳如何网上全流程注册公司?  在线教育网站制作平台,山西立德教育官网?  如何配置WinSCP新建站点的密钥验证步骤?  如何高效利用亚马逊云主机搭建企业网站?  5种Android数据存储方式汇总  微信小程序制作网站有哪些,微信小程序需要做网站吗?  淘宝制作网站有哪些,淘宝网官网主页?  如何通过商城免费建站系统源码自定义网站主题?  建站之星如何修改网站生成路径?  建站为何优先选择香港服务器?  实现点击下箭头变上箭头来回切换的两种方法【推荐】  Swift中循环语句中的转移语句 break 和 continue  邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?  建站与域名管理如何高效结合?  如何在建站之星绑定自定义域名?  C++如何使用std::optional?(处理可选值)  重庆网站制作公司哪家好,重庆中考招生办官方网站?  个人摄影网站制作流程,摄影爱好者都去什么网站?  建站IDE高效指南:快速搭建+SEO优化+自适应模板全解析  制作充值网站的软件,做人力招聘为什么要自己交端口钱?  小视频制作网站有哪些,有什么看国内小视频的网站,求推荐?  如何安全更换建站之星模板并保留数据?  家具网站制作软件,家具厂怎么跑业务?  佛山网站制作系统,佛山企业变更地址网上办理步骤?  如何解决ASP生成WAP建站中文乱码问题?  金*站制作公司有哪些,金华教育集团官网?  广州网站建站公司选择指南:建站流程与SEO优化关键词解析  如何快速使用云服务器搭建个人网站?  h5在线制作网站电脑版下载,h5网页制作软件?  微信网站制作公司有哪些,民生银行办理公司开户怎么在微信网页上查询进度?  建站之星安装步骤有哪些常见问题?  浅谈Javascript中的Label语句  如何快速辨别茅台真假?关键步骤解析  建站主机默认首页配置指南:核心功能与访问路径优化  宝塔建站教程:一键部署配置流程与SEO优化实战指南  c++怎么使用类型萃取type_traits_c++ 模板元编程类型判断【方法】  ui设计制作网站有哪些,手机UI设计网址吗? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。