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),对他们不进行压缩处理
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。