源码网商城,靠谱的源码在线交易网站 我的订单 购物车 帮助

源码网商城

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

  • 时间:2020-01-13 07:30 编辑: 来源: 阅读:
  • 扫一扫,手机访问
摘要: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,否则不会持续输出到客户端,也就不会看到连续的进度条变化。 [b]总结: [/b] 上面的部分比较乱,我这里总结一下关键点。 1、在上传组件里面,把总大小和当前读取了的大小放到一个类里面,并持续更新,直到处理完毕 2、上传的进度类,放在session里面,供进度读取页面读取 3、进度读取页面,从session里面拿到数据,并返回结果。 有几个疑问解释一下。 1、由于Http协议决定了,必须等request处理完毕才会返回输出,所以不能在upload页面里进行处理进度的显示。我前面测试到1M左右的文件不成功,就是没有考虑到这个问题。所以必须单独用一个GET的程序进行读取 2、读取是一个持续不断的过程,因为上传大文件是很慢的! 3、如果你的应用服务器启用了GZIP压缩,是容器管理的,那么很不幸,因为容易必须拿到所有的数据,至少是一部分数据才会返回,所以造成我们返回的那些很少的字节经常会被截住,造成无法显示上传的连续过程。    [b]解决方法  [/b]  1) 关闭GZIP, 我想许多人不会这么做    2) 使用自定义的GZIP压缩,判断某些东西(比如URL),对他们不进行压缩处理 以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程素材网。
  • 全部评论(0)
联系客服
客服电话:
400-000-3129
微信版

扫一扫进微信版
返回顶部