梦幻e家人

java咖啡
随笔 - 15, 文章 - 0, 评论 - 11, 引用 - 0
数据加载中……

掌控上传进度的AJAX Upload

掌控上传进度的AJAX Upload

cleverpig 发表于 2007-01-08 11:12:14
作者:cleverpig     来源:Matrix
评论数:83 点击数:5,066     投票总得分:12 投票总人次:4
关键字:AJAX,upload,monitor

摘要:

2006年底Google了一下AJAX Upload实现,结果没有发现很完整的Java实现。硕果仅存的就是TELIO公司的Pierre-Alexandre发表的《AJAX Upload progress monitor for Commons-FileUpload Example》一文。虽然文中完成Upload工作的是Common-FileUpload组件,但在其代码中没有1.2版本所提供的Listener功能,这就对检测文件上传情况造成了困难...
掌控上传进度的AJAX Upload

作者:cleverpig

image
AJAX——最酷的“冲浪板”



动机:

        2006年底Google了一下AJAX Upload实现,结果没有发现很完整的Java实现。硕果仅存的就是TELIO公司的Pierre-Alexandre发表的《AJAX Upload progress monitor for Commons-FileUpload Example》文中提供的ajax-upload-1.0.war

        虽然上文中完成Upload工作的是Apache的Common-FileUpload组件,但在其代码中所使用的FileUpload1.1版本并没有1.2版本所提供的上传处理Listener功能,这就对检测文件上传情况造成了困难。我想正是这个原因致使Pierre-Alexandre使用了DWR+MonitoredDiskFileItem、MonitoredDiskFileItemFactory类(分别继承DiskFileItemDiskFileItemFactory)的方式:前者负责在web客户端进行Remote Call;后者在进行文件数据读取时统计数据总量、读取数据量、处理文件总数,并保存于Session中,以供web客户端通过DWR远程调用UploadMonitor类的getUploadInfo方法进行轮询(Poll)。

        从本人观点出发,Pierre-Alexandre实现的不足之处:
        1.没有用户取消上传功能;
        2.完全的DWR实现,没有使用Prototype,对于不会使用DWR的开发者来讲有一定的知识局限性,而且由于DWR的个性而造成不便将此实现集成到项目中。



Prototype+Servlet的实现:

image
Prototype+Servlet的Example


        所以出于研究Prototype之目的,本人经过仔细思考,尝试实现了一个Prototype+Servlet的简单Example。其工作流程很简单:
1.在Form提交上传文件Field的同时,使用AJAX周期性地从Servlet轮询上传状态信息;
2.然后,根据此信息更新进度条和相关文字,及时反映文件传输状态;
3.如果用户取消上传操作,则进行相应的现场清理工作:删除已经上传的文件,在Form提交页面中显示相关信息;
4.如果上传完毕,在Form提交页面中显示已经上传的文件内容(或链接),也可以与一些AJAX SlideShow应用结合在一起。

服务器端代码:

        Bean序列化/反序列化工作:XmlUnSerializer这个类虽然不能够通吃任何模样的Bean,但应付一般的Bean、具有Collection类型属性的Bean和Bean List来讲还是够用的。
        {XmlUnSerializer类的核心方法serializeBean和serializeBeanList}:

        /**
         * 将bean系列化为UTF-8编码的xml
         * @param beanObj
         * @return
         * @throws IOException
         */
        public static String serializeBean(Object beanObj) throws IOException{
        …
        }
        /**
         * 将bean列表序列化为UTF-8编码的xml
         * @param beanObj
         * @return
         * @throws IOException
         */
        public static String serializeBeanList(Object beanListObj) throws IOException{
        …
        }


        文件上传状态Bean:使用FileUploadStatus这个类记录文件上传状态,并将其作为服务器端与web客户端之间通信的媒介物:通过对这个类对象进行XML序列化作为服务器回应发送给web客户端,web客户端使用JavaScript对其进行反序列化处理获得JavaScript版本的文件上传状态对象。
        {FileUploadStatus的属性}:

        //上传总量
        private long uploadTotalSize=0;
        //读取上传总量
        private long readTotalSize=0;
        //当前上传文件号
        private int currentUploadFileNum=0;
        //成功读取上传文件数
        private int successUploadFileCount=0;
        //状态
        private String status="";
        //处理起始时间
        private long processStartTime=0l;
        //处理终止时间
        private long processEndTime=0l;
        //处理执行时间
        private long processRunningTime=0l;
        //上传文件URL列表
        private List uploadFileUrlList=new ArrayList();
        //取消上传
        private boolean cancel=false;
        //上传base目录
        private String baseDir="";


        文件上传状态监视工作:使用Common-FileUpload 1.2版本(20070103)。此版本与1.1版的区别在于提供了能够监视文件上传情况的ProcessListener接口,使开发者通过FileUploadBase类对象的setProcessListener方法植入自己的Listener,而且实现这个Listener很简单
        {FileUploadListener主要方法update}:

        /**
         * 更新状态
         * @param pBytesRead 读取字节总数
         * @param pContentLength 数据总长度
         * @param pItems 当前正在被读取的field号
         */
        public void update(long pBytesRead, long pContentLength, int pItems){
                FileUploadStatus fuploadStatus=BackGroundService.takeOutFileUploadStatusBean(this.session);
                logger.debug("当前正在处理第" + pItems+"个文件");
                fuploadStatus.setUploadTotalSize(pContentLength);
                //读取完成
            if (pContentLength == -1) {
               logger.debug("读取完成:读取了 " + pBytesRead + " bytes.");
               fuploadStatus.setStatus("完成对" + pItems+"个文件的读取:读取了 " + pBytesRead + " bytes.");
               fuploadStatus.setReadTotalSize(pBytesRead);
               fuploadStatus.setSuccessUploadFileCount(pItems);
               fuploadStatus.setProcessEndTime(System.currentTimeMillis());
               fuploadStatus.setProcessRunningTime(fuploadStatus.getProcessEndTime());
            //读取中
            } else {
               logger.debug("读取进行中:已经读取了 " + pBytesRead + " / " + pContentLength+ " bytes.");
               fuploadStatus.setStatus("当前正在处理第" + pItems+"个文件:已经读取了 " + pBytesRead + " / " + pContentLength+ " bytes.");
               fuploadStatus.setReadTotalSize(pBytesRead);
               fuploadStatus.setCurrentUploadFileNum(pItems);
               fuploadStatus.setProcessRunningTime(System.currentTimeMillis());
            }
            BackGroundService.storeFileUploadStatusBean(this.session,fuploadStatus);
        }

        很清楚,我也把FileUploadStatus这个Bean存取于Session中。

        Servlet实现:BackGroundService这个Servlet类负责接收Form Post数据、回应状态轮询请求、处理取消文件上传的请求。尽管可以把这些功能相互分离开来(比如构造一个FileUploadManager类),但出于简单明了、便于阅读之目的,还是将它们放到Servlet中,只是由不同的方法进行分割。
        {BackGroundService中的processFileUpload方法用于处理文件上传请求}:

        /**
         * 处理文件上传
         * @param request
         * @param response
         * @throws IOException
         * @throws ServletException
         */
        private void processFileUpload(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException{
                DiskFileItemFactory factory = new DiskFileItemFactory();
                //设置内存阀值,超过后写入临时文件
                factory.setSizeThreshold(10240000);
                //设置临时文件存储位置
                factory.setRepository(new File(request.getRealPath("/upload/temp")));
                ServletFileUpload upload = new ServletFileUpload(factory);
                //设置单个文件的最大上传size
                upload.setFileSizeMax(10240000);
                //设置整个request的最大size
                upload.setSizeMax(10240000);
                upload.setProgressListener(new FileUploadListener(request.getSession()));
                //保存初始化后的FileUploadStatus Bean
                storeFileUploadStatusBean(request.getSession(),initFileUploadStatusBean(request));
                
                String forwardURL="";
                try {
                        List items = upload.parseRequest(request);
                        //获得返回url
                        for(int i=0;i<items.size();i++){
                                FileItem item=(FileItem)items.get(i);
                                if (item.isFormField()){
                                        logger.debug("form Field["+item.getFieldName()+"]="+item.getString());
                                        forwardURL=item.getString();
                                        break;
                                }
                        }
                        //处理文件上传
                        for(int i=0;i<items.size();i++){
                                FileItem item=(FileItem)items.get(i);

                                //取消上传
                                if (takeOutFileUploadStatusBean(request.getSession()).getCancel()){
                                        deleteUploadedFile(request);
                                        break;
                                }
                                //保存文件
                                else if (!item.isFormField() && item.getName().length()>0){
                                        String fileName=takeOutFileName(item.getName());
                                        logger.debug("处理文件["+fileName+"]:保存路径为"
                                                        +request.getRealPath(UPLOAD_DIR)+File.separator+fileName);
                                        File uploadedFile = new File(request.getRealPath(UPLOAD_DIR)+File.separator+fileName);
                                        item.write(uploadedFile);
                                        //更新上传文件列表
                                        FileUploadStatus fUploadStatus=takeOutFileUploadStatusBean(request.getSession());
                                        fUploadStatus.getUploadFileUrlList().add(fileName);
                                        storeFileUploadStatusBean(request.getSession(),fUploadStatus);
                                        Thread.sleep(500);
                                }
                        }
                
                } catch (FileUploadException e) {
                        logger.error("上传文件时发生错误:"+e.getMessage());
                        e.printStackTrace();
                        uploadExceptionHandle(request,"上传文件时发生错误:"+e.getMessage());
                } catch (Exception e) {
                        // TODO Auto-generated catch block
                        logger.error("保存上传文件时发生错误:"+e.getMessage());
                        e.printStackTrace();
                        uploadExceptionHandle(request,"保存上传文件时发生错误:"+e.getMessage());
                }
                if (forwardURL.length()==0){
                        forwardURL=DEFAULT_UPLOAD_FAILURE_URL;
                }
                request.getRequestDispatcher(forwardURL).forward(request,response);
        }


        {BackGroundService中的responseFileUploadStatusPoll方法用于处理对文件上传状态的轮询请求}:

        /**
         * 回应上传状态查询
         * @param request
         * @param response
         * @throws IOException
         */
        private void responseFileUploadStatusPoll(HttpServletRequest request,HttpServletResponse response) throws IOException{
                response.setContentType("text/xml");
                response.setCharacterEncoding("UTF-8");
                response.setHeader("Cache-Control", "no-cache");
                logger.debug("发送上传状态回应");
                response.getWriter().write(XmlUnSerializer.serializeBean(
                                request.getSession().getAttribute(UPLOAD_STATUS)));
        }


        {BackGroundService中的processCancelFileUpload方法用于处理取消文件上传的请求}:

        /**
         * 处理取消文件上传
         * @param request
         * @param response
         * @throws IOException
         */
        private void processCancelFileUpload(HttpServletRequest request,HttpServletResponse response) throws IOException{
                FileUploadStatus fUploadStatus=(FileUploadStatus)request.getSession().getAttribute(UPLOAD_STATUS);
                fUploadStatus.setCancel(true);
                request.getSession().setAttribute(UPLOAD_STATUS, fUploadStatus);
                responseFileUploadStatusPoll(request,response);
        }


Web客户端代码:

image
Prototype给开发者更多的自由选择


web客户端使用了基于Prototype的AjaxWrapper类和XMLDomForAjax类,前者实现了对Ajax.Request功能的封装,而后者实现了对来自服务器的XML Response的反序列化(反序列化为JavaScript对象)。

        为了避免在AjaxWrapper的回调方法中发生this被重写的问题,我使用了ClassUtils类给任何类的每个方法注册一个对类对象自身引用,详见《解开JavaScript生命的达芬奇密码》《Prototype.AjaxRequest的调用堆栈重写问题》
        {ClassUtils类代码}:

//类工具
var ClassUtils=Class.create();
ClassUtils.prototype={
        _ClassUtilsName:'ClassUtils',
        initialize:function(){
        },
        /**
         * 给类的每个方法注册一个对类对象的自我引用
         * @param reference 对类对象的引用
         */
        registerFuncSelfLink:function(reference){
                for (var n in reference) {
                var item = reference[n];                        
                if (item instanceof Function)
                                item.$ = reference;
            }
        }
}


        {将XML反序列化为JavaScript对象的XMLDomForAjax类代码}:

var XMLDomForAjax=Class.create();
XMLDomForAjax.prototype={
        isDebug:false,
        //dom节点类型常量
        ELEMENT_NODE:1,
        ATTRIBUTE_NODE:2,
    TEXT_NODE:3,
    CDATA_SECTION_NODE:4,
    ENTITY_REFERENCE_NODE:5,
    ENTITY_NODE:6,
    PROCESSING_INSTRUCTION_NODE:7,
    COMMENT_NODE:8,
    DOCUMENT_NODE:9,
    DOCUMENT_TYPE_NODE:10,
    DOCUMENT_FRAGMENT_NODE:11,
    NOTATION_NODE:12,
    
        initialize:function(isDebug){
                new ClassUtils().registerFuncSelfLink(this);
                this.isDebug=isDebug;
        },
        /**
         * 建立跨平台的dom解析器
         * @param xml xml字符串
         * @return dom解析器
         */
        createDomParser:function(xml){
                // code for IE
                if (window.ActiveXObject){
                  var doc=new ActiveXObject("Microsoft.XMLDOM");
                  doc.async="false";
                  doc.loadXML(xml);
                }
                // code for Mozilla, Firefox, Opera, etc.
                else{
                  var parser=new DOMParser();
                  var doc=parser.parseFromString(xml,"text/xml");
                }
                return doc;
        },
        /**
         * 反向序列化xml到javascript Bean
         * @param xml xml字符串
         * @return javascript Bean
         */
        deserializedBeanFromXML:function (xml){
                var funcHolder=arguments.callee.$;
                var doc=funcHolder.createDomParser(xml);
                // documentElement总表示文档的root
                var objDomTree=doc.documentElement;
                var obj=new Object();
            for (var i=0; i<objDomTree.childNodes.length; i++) {
                    //获得节点
                    var node=objDomTree.childNodes[i];
                    //取出其中的field元素进行处理
                if ((node.nodeType==funcHolder.ELEMENT_NODE) && (node.tagName == 'field')) {
                        var nodeText=funcHolder.getNodeText(node);
                        if (funcHolder.isDebug){
                            alert(node.getAttribute('name')+' type:'+node.getAttribute('type')+' text:'+nodeText);
                        }
                    var objFieldValue=null;
                    //如果为列表
                    if (node.getAttribute('type')=='java.util.List'){
                            if (objFieldValue && typeof(objFieldValue)=='Array'){
                                    if (nodeText.length>0){
                                                        objFieldValue[objFieldValue.length]=nodeText;
                                                }
                                        }
                                        else{
                                                objFieldValue=new Array();
                                        }
                                }
                                else if (node.getAttribute('type')=='long'
                                        || node.getAttribute('type')=='java.lang.Long'
                                        || node.getAttribute('type')=='int'
                                        || node.getAttribute('type')=='java.lang.Integer'){
                                        
                                        objFieldValue=parseInt(nodeText);
                                }
                                else if (node.getAttribute('type')=='double'
                                        || node.getAttribute('type')=='float'
                                        || node.getAttribute('type')=='java.lang.Double'
                                        || node.getAttribute('type')=='java.lang.Float'){
                                        
                                        objFieldValue=parseFloat(nodeText);
                                }
                                else if (node.getAttribute('type')=='java.lang.String'){
                                        objFieldValue=nodeText;
                                }
                                else{
                                        objFieldValue=nodeText;
                                }
                                //赋值给对象
                                obj[node.getAttribute('name')]=objFieldValue;
                                if (funcHolder.isDebug){
                                        alert(eval('obj.'+node.getAttribute('name')));
                                }
                }
                else if (node.nodeType == funcHolder.TEXT_NODE){
                        if (funcHolder.isDebug){
                                //alert('TEXT_NODE');
                        }
                        
                }
                else if (node.nodeType == funcHolder.CDATA_SECTION_NODE){
                        if (funcHolder.isDebug){
                                //alert('CDATA_SECTION_NODE');
                        }
                }
            }
            return obj;
        },
        /**
         * 获得dom节点的text
         */
        getNodeText:function (node) {
                var funcHolder=arguments.callee.$;
            // is this a text or CDATA node?
            if (node.nodeType == funcHolder.TEXT_NODE || node.nodeType == funcHolder.CDATA_SECTION_NODE) {
                return node.data;
            }
            var i;
            var returnValue = [];
            for (i = 0; i < node.childNodes.length; i++) {
                    //采用递归算法
                returnValue.push(funcHolder.getNodeText(node.childNodes[i]));
            }
            return returnValue.join('');
        }
}


        {AjaxWrapper类的主要方法putRequest和callBackHandler}:

         /**
         * 以get的方式向server发送request
         * @param url
         * @param params
         * @param callBackFunction 发送成功后回调的函数或者函数名
         */
        putRequest:function(url,params,callBackFunction){
                var funcHolder=arguments.callee.$;
            var xmlHttp = new Ajax.Request(url,
                        {
                                method: 'get',
                            parameters: params,
                                requestHeaders:['my-header-encoding','utf-8'],
                            onFailure: function(){
                                        alert('对不起,网络通讯失败,请重新刷新!');
                                },
                                onSuccess: function(transport){
                                },
                                onComplete: function(transport){
                                        funcHolder.callBackHandler.apply(funcHolder,[transport,callBackFunction]);
                                }
                        });
        },
        /**
         * 远程调用的回调处理
         * @param transport xmlhttp的transport
         * @param callBackFunction 回调时call的方法,可以是函数也可以是函数名
         */
        callBackHandler:function(transport,callBackFunction){
                var funcHolder=arguments.callee.$;
                if(transport.status!=200){
                        alert("获得回应失败,请求状态:"+transport.status);
                }
                else{
                        funcHolder.xml_source=transport.responseText;
                        if (funcHolder.debug_flag)
                                alert('call callback function');
                        if (typeof(callBackFunction)=='function'){
                                if (funcHolder.debug_flag){
                                        alert('invoke callbackFunc');
                                }
                                callBackFunction(transport.responseText);
                        }
                        else{
                                if (funcHolder.debug_flag){
                                        alert('evalFunc callbackFunc');
                                }
                                new execute().evalFunc(callBackFunction,transport.responseText);
                        }
                        if (funcHolder.debug_flag)
                                alert('end callback function');
                }
        }


        {页面中主要的JavaScript方法:refreshUploadStatus和startProcess/cancelProcess}:

//刷新上传状态
function refreshUploadStatus(){
        var ajaxW = new AjaxWrapper(false);
        ajaxW.putRequest(
                './uploadStatus.action',
                'uploadStatus=',
                function(responseText){
                        var deserialor=new XMLDomForAjax(false);
                        var uploadInfo=deserialor.deserializedBeanFromXML(responseText);
                        var progressPercent = Math.ceil(
                                (uploadInfo.readTotalSize) / uploadInfo.uploadTotalSize * 100);

                $('progressBarText').innerHTML = ' 上传处理进度: '+progressPercent+'% ['+
                        (uploadInfo.readTotalSize)+'/'+uploadInfo.uploadTotalSize + ' bytes]'+
                        ' 正在处理第'+uploadInfo.currentUploadFileNum+'个文件'+
                        ' 耗时: '+(uploadInfo.processRunningTime-uploadInfo.processStartTime)+' ms';
                $('progressStatusText').innerHTML=' 反馈状态: '+uploadInfo.status;
                $('totalProgressBarBoxContent').style.width = parseInt(progressPercent * 3.5) + 'px';
                }
        );
}
//上传处理
function startProgress(){
        Element.show('progressBar');
    $('progressBarText').innerHTML = ' 上传处理进度: 0%';
    $('progressStatusText').innerHTML=' 反馈状态:';
    $('uploadButton').disabled = true;
    var periodicalExe=new PeriodicalExecuter(refreshUploadStatus,2);
    return true;
}
//取消上传处理
function cancelProgress(){
        $('cancelUploadButton').disabled = true;
        var ajaxW = new AjaxWrapper(false);
        ajaxW.putRequest(
                './uploadStatus.action',
                'cancelUpload=true',
                //因为form的提交,这可能不会执行
                function(responseText){
                        var deserialor=new XMLDomForAjax(false);
                        var uploadInfo=deserialor.deserializedBeanFromXML(responseText);
                        $('progressStatusText').innerHTML=' 反馈状态: '+uploadInfo.status;
                        if (msgInfo.cancel=='true'){
                                alert('删除成功!');
                                window.location.reload();
                        };
                }
        );
}


运行界面:

image
起始页面



image
上传进行中…



image
上传完成后的文件列表



image
用户取消上传后显示的页面




image
上传过程中出错(上传文件过大)页面


源代码下载:






相关链接:
        AJAX Upload progress monitor for Commons-FileUpload Example
        Apache Common FileUpload组件
        Prototype官方网站
        IBM的AJAX SlideShow应用
        解开JavaScript生命的达芬奇密码
        Prototype.AjaxRequest的调用堆栈重写问题

感谢阅读此文

        请支持cleverpig发起的image


本页页面地址:

投票评分(记入本贴作者的专家分)

     非常好 还行 一般 扔鸡蛋          投票总得分: / 投票总人次:

用户评论列表

#1 评论作者: GOVO 发表时间: 2007-01-08 01:23 下午

这个方法早想到了,就等common.FileUpload的1.2出来。

#2 评论作者: GOVO 发表时间: 2007-01-08 01:24 下午

让老外先发表,我晕哦~

#3 评论作者: cleverpig 发表时间: 2007-01-08 02:08 下午

GOVO,这是cleverpig的原创,可不是老外的作品哦。

#4 评论作者: jctr 发表时间: 2007-01-09 02:01 下午

import com.bjinfotech.util.objecttk.*;
没有,程序无法运行

#5 评论作者: mreay 发表时间: 2007-01-09 02:13 下午

貌似不错,抽空试试!!!
有人用AS实现过类似的功能吗?

#6 评论作者: cleverpig 发表时间: 2007-01-09 05:49 下午

to jctr:感谢你的纠错!我已经修改了上面的源代码下载:直接提供war包,源代码在WEB-INF/classes目录中。

#7 评论作者: qhz 发表时间: 2007-01-10 09:02 上午

正好用上,赶紧试试!

#8 评论作者: archer1207 发表时间: 2007-01-10 09:29 上午

其实ss有想类似的实现:http://wiki.springside.org.cn/display/springside/AjaxUpload
不过作者的也不错,+U.

#9 评论作者: cleverpig 发表时间: 2007-01-10 10:34 上午

springside提供的AjaxUpload采用DWR+MVC框架的,而我提供的是比较original的版本,出于简化的目的单纯地使用prototype+jsp,这样不和任何框架耦合,适用于任何支持Sevlet的java web框架。

#10 评论作者: haidaotao 发表时间: 2007-01-11 10:33 上午

在tomcat里上传中文文件名我最头疼了,传是传的上去,但是点击后是404错误。不知道AJAX Upload有么有办法解决

#11 评论作者: kq1983 发表时间: 2007-01-11 03:28 下午

     tomcat重起后,第一次上传没有进度条(连接失败,500,后台在跑),第二次上传以后才开始出现进度条,cleverpig这问题怎么解决.

#12 评论作者: kq1983 发表时间: 2007-01-11 03:50 下午

      浏览器关闭后,第一次就是不会出现进度条(连接失败,500,后台在跑),第二次开始才出来进入条.

#13 评论作者: easyrun 发表时间: 2007-01-11 03:51 下午

还有一个问题,文件上传时文件的索引不正确,总是从开始处理第2个文件开始,就算是上传一个文件也是显示正在处理第2个文件

#14 评论作者: easyrun 发表时间: 2007-01-11 03:58 下午

连接失败,500,这个问题我也遇到过,莫名其妙的有好了。
问题出现原因是在返回状态给客户端时,到session里去取UPLOAD_STATUS没有取到,然后再调用serializeBean方法时出现nullpointexception。
因为在执行
Field[] fields=beanObj.getClass().getDeclaredFields();
时beanObj为空。
建议可以在这里做个保护措施。如果为空怎么怎么样。

不过session里的UPLOAD_STATUS为什么为空这个原因还没有找到

#15 评论作者: easyrun 发表时间: 2007-01-11 04:18 下午

取消上传的功能对小文件来说根本没有意义,因为上传小文件时基本上是一闪就过,没有机会来取消,呵呵。
是否可以把取消下载这个动作放到所有文件上传后再让用户来选择取消,如果可以的话能做到在上传多文件时可以取消其中的任意一个上传文件那是最好了。
观察了一下取消下载动作本身就是在所有文件上传完了之后才生效的,没有细究上面的想法在实现上是否可行,只是觉得那样可能会更方便一点。

#16 评论作者: softicer 发表时间: 2007-01-12 01:08 上午

谢谢 cleverpig  的原创。难得找到中文的参考资料了,呵呵。

#17 评论作者: cleverpig 发表时间: 2007-01-12 04:37 下午

根据大家的反馈,已经完成了AJAX FileUpload的u1版,进行了以下fix:
1。支持中文文件下载
2。增加了单个文件的删除功能
3。增加了AJAX动画显示
大家可以从本文的源代码处或者这里下载

#18 评论作者: cleverpig 发表时间: 2007-01-12 04:39 下午

to haidaotao同学:

>>>在tomcat里上传中文文件名我最头疼了,传是传的上去,但是点击后是404错误。不知道AJAX Upload有么有办法解决

u1版已经使用servlet处理文件下载请求(使用PlainURLEncoder类对文件名进行了编解码),而不是从前的直接链接。

#19 评论作者: cleverpig 发表时间: 2007-01-12 04:42 下午

to  kq1983同学:
关于进度条的显示问题,完全依靠了AJAX的调用周期长短(目前代码中为2秒)。所以也许在2秒内第一个文件已经被处理完,而AJAX反馈从server回来便会显示正在处理第2个文件。

#20 评论作者: cleverpig 发表时间: 2007-01-12 04:49 下午

to easyrun同学:

关于处理进度的问题请参考上一条。

>>是否可以把取消下载这个动作放到所有文件上传后再让用户来选择取消,如果可以的话能做到在上传多文件时可以取消其中的任意一个上传文件那是最好了。

u1版已经增加上传后的删除功能。而在上传多文件时的取消处理时,由于使用fileUpload的解析文件功能无法中断,所以我是在解析文件后的保存文件的循环中完成的——删除所有已经上传的文件、清空session中的文件上传列表。我想这就是出现你所见到的“文件上传后才删除”的原因。

#21 评论作者: zengxianhong 发表时间: 2007-01-12 06:15 下午

浏览器关闭后,第一次就是不会出现进度条(连接失败,500,后台在跑),第二次开始才出来进入条是因为processFileUpload方法中的request.getSession()取得session和responseFileUploadStatusPoll方法中的request.getSession()取得session不同。

#22 评论作者: cleverpig 发表时间: 2007-01-12 09:58 下午

to zengxianhong 同学:

在浏览器关闭后,原来的回话将结束(在server中也将如此)。这样在下次打开fileupload.html进行上传时,便会建立新的session,而在processFileUpload方法和responseFileUploadStatusPoll方法中的session都来自于同一个client的session。而且如果起初发生连接失败(500),那证明后天可能发生错误(建议使用FireFox的firebug查看一下XMLHTTPResponse的内容)。

#23 评论作者: baiyun 发表时间: 2007-01-12 11:46 下午

这样用Session存取UPLOAD_STATUS的话,因为这个key确定,所以同一时间是能有一个页面进行上传,要是我想多个页面同时执行上传动作,请问应该怎么样修改?就是js端怎么能动态的得到这个Session的key?

#24 评论作者: discover7 发表时间: 2007-01-13 01:41 下午

这个东东能不能运行啊。
先试试看

#25 评论作者: cleverpig 发表时间: 2007-01-14 10:46 上午

to baiyun同学:这需要增加一个server端的seesionManager功能,对session进行管理,比如firefox的浏览器tab会导致多个页面共享同一个session,而不是IE那样的1 session/page。

#26 评论作者: zhaow8810 发表时间: 2007-01-15 08:32 下午

不用session用cookie存储上传状态能实现吗

#27 评论作者: cleverpig 发表时间: 2007-01-15 09:08 下午

to zhaow8810同学:
>>不用session用cookie存储上传状态能实现吗?
cookie并不具有数据存储的独立性、隔离性,而session是基于浏览器单次会话的存储体。所以为了将上传数据在server与client之间共享,session是很好的媒介。
但是有些浏览器对session的share行为,使得单纯使用session并不能达到精准的数据隔离,因此在下一个版本中将增加SessionManger。

#28 评论作者: eidff 发表时间: 2007-01-15 09:35 下午

为什么我的运行时,虽然报错也可以上传文件成功?

#29 评论作者: zhaow8810 发表时间: 2007-01-16 12:19 上午

那我的系统就没用session保存用户状态啊 在集群体环境下为了多服务器更好的性能而用cookie保存用户状态,这样减少了session序列化给服务器的压力。而在这种环境下能否集成您提供的ajax上传文件功能那?

#30 评论作者: zengxianhong 发表时间: 2007-01-16 08:02 上午

to cleverpig
第一次部署运行时,文件上传成功,但是从服务器端得到的消息是500的错误。我debug了一下,发现第一次运行时,processFileUpload方法和responseFileUploadStatusPoll方法中取得的session不是同一个session。第二次之后就没问题了。

#31 评论作者: cleverpig 发表时间: 2007-01-16 09:07 上午

to zhaow8810同学:
>>那我的系统就没用session保存用户状态啊 在集群体环境下为了多服务器更好的性能而用cookie保存用户状态,这样减少了session序列化给服务器的压力。

用户状态保存在client也可以,但是建议你使用session cookie。使用session cookie保存用户信息的话,可以替代server端对session的存取管理,用AJAX将处理状态传送到client,进而保存到session cookie。
推荐你一个简单的session cookie实现

#32 评论作者: cleverpig 发表时间: 2007-01-16 09:10 上午

to zengxianhong同学:
>>第一次部署运行时,文件上传成功,但是从服务器端得到的消息是500的错误。我debug了一下,发现第一次运行时...

这个问题我测试时没有发现,可能和你我的运行环境差异有关:web容器、servlet api version、浏览器版本等。

#33 评论作者: zengxianhong 发表时间: 2007-01-16 09:26 上午

to cleverpig
我的web容器:apache-tomcat-5.5.17
servlet api version:2.4
浏览器版本: firefox_2.0.0.1
                 IE_6.0.2900
两个浏览器我都测了,都有那种现象。

#34 评论作者: liusf 发表时间: 2007-01-16 02:35 下午

讨论相当激烈啊....感谢分享...确实不错..

#35 评论作者: sljsuper 发表时间: 2007-01-16 03:08 下午

我也出现 zengxianhong的问题!!删除的时候也是!

#36 评论作者: agu 发表时间: 2007-01-16 03:48 下午

我也遇到同样的问题 删文件时报错

java.lang.NoClassDefFoundError: org/apache/commons/collections/FastHashMap
        at org.apache.commons.beanutils.ConvertUtilsBean.<init>(ConvertUtilsBean
.java:125)

#37 评论作者: zengxianhong 发表时间: 2007-01-16 05:28 下午

to agu
上面的错误是缺少commons-collections.jar,和session没有关系。

#38 评论作者: 191301587 发表时间: 2007-01-18 01:18 下午

无法下载...............

#39 评论作者: 191301587 发表时间: 2007-01-18 01:19 下午

楼主或哪位能否给我发一个学习学习,谢谢

ivanw@126.com

#40 评论作者: haiter 发表时间: 2007-01-19 11:15 上午

当上传文件为100M时,在上传的过程中,点击取消。实际上服务器上并没有取消。如何才能真正取消?

#41 评论作者: cleverpig 发表时间: 2007-01-19 03:52 下午

to zengxianhong同学:
>>第一次部署运行时,文件上传成功,但是从服务器端得到的消息是500的错误。我debug了一下,发现第一次运行时...

问题已经找到,在某些情况下的确出现ajax xmlHttpRequest和form post的session不一致的现象。

#42 评论作者: cleverpig 发表时间: 2007-01-19 03:53 下午

针对上面的问题,我发布u2版本,进行了部分改进和提升。

#43 评论作者: cleverpig 发表时间: 2007-01-19 03:54 下午

U2's new Feature:
1.使用独立的UploadSessionManager管理uploadSession,使后者在AJAX和form之间共享。分离原处一处的“处理文件上传”、“状态查询”任务到UploadProcessService和FileUploadCommandService这两个servlet中,实现上传和查询的操作分离,提高反应速度。
2.使用AJAX join命令,首先发送AJAX请求,然后进行form submit,这样可以达到session的一致;
3.增加了MyServletFileUpload类,继承与ServletFileUpload,并在其parserRequest方法中增加了cancel upload处理;
4.增加在servlet进行处理前的上传文件超限的检测能力;
5.upload和result页面完全使用AJAX+DHTML,摆脱了对JSP的依赖。

#44 评论作者: cleverpig 发表时间: 2007-01-19 04:35 下午

to haiter同学:
>>当上传文件为100M时,在上传的过程中,点击取消。实际上服务器上并没有取消。如何才能真正取消?

我没有想出实现“真正的取消”的办法,因为form submit是事务性的,没法中间取消掉。所以即使后台已经执行玩取消操作了,但form仍然要提交完才行。

#45 评论作者: zengxianhong 发表时间: 2007-01-19 04:51 下午

to cleverpig
知道是什么原因引起的ajax xmlHttpRequest和form post的session不一致的现象吗?

#46 评论作者: haiter 发表时间: 2007-01-19 05:09 下午

当上传文件为100M时,在上传的过程中,点击取消。服务器上还在不断地打印如下信息:说明没有真正取消。如何才能真正取消?
----
[hirycajulyrosuvydyso]被添加/更新
17:06:51,515 DEBUG FileUploadStatusBeanManager:20 - 使用已建立的UploadSession[hirycajulyrosuvydyso]


#47 评论作者: cleverpig 发表时间: 2007-01-20 09:08 下午

to haiter同学:

>>服务器上还在不断地打印如下信息:说明没有真正取消。如何才能真正取消?

这些信息表示uploadsession中的fileUploadStatus被更新,因为已经执行了取消,所以要更新fileUploadStatus中的信息。虽然终止了对form提交的request的解析处理工作,但web 容器对form提交数据的处理仍然要完成。目前没找到合适的方法实现真正的取消。

#48 评论作者: cleverpig 发表时间: 2007-01-20 09:13 下午

to zengxianhong:
>>知道是什么原因引起的ajax xmlHttpRequest和form post的session不一致的现象吗?

原因还没有找到,但问题现象很容易再现,只需servlet和AJAX、form就可以。
1。首先使用form提交上传文件(文件足够大需要执行一段时间),然后发送AJAX请求到servlet,servlet将分配给两种request不同的session。
2。首先发送AJAX请求,然后再提交form数据,servlet将分配给第一次的AJAX请求一个新session,而后的form请求将重用这个session。

#49 评论作者: chenlb 发表时间: 2007-01-21 04:14 下午

u2版,要在lib下加commons-collections.jar

上传的进度不能反应出来,轮查时间间隔太短了,有时一部到100%
有时只有2次变化,

我本地上传3M左右的文件,只有2-3次的变化, 建议进度条进度变化做细的点.

感谢你的分享

#50 评论作者: cleverpig 发表时间: 2007-01-22 09:28 上午

to chenlb同学:

>>上传的进度不能反应出来,轮查时间间隔太短了,有时一部到100%
有时只有2次变化

轮询时间可以自定义,只要修改fileupload.html页面中函数startProgress中:


var periodicalExe=new PeriodicalExecuter(refreshUploadStatus,5);

其中第二个参数是轮询周期,目前默认是5秒。

 

#51 评论作者: cleverpig 发表时间: 2007-01-22 09:48 上午

to 缺少commons-collections.jar的同学们:

这个问题将在下一个update版本中修复。
可以到这里下载本人使用的3.1版:

,或者下载更新的版本,把它放入到WEB-INF/lib目录下即可。

 

#52 评论作者: cleverpig 发表时间: 2007-01-30 06:10 下午

更新到U3版本:
U3 part1
U3 part2

#53 评论作者: cleverpig 发表时间: 2007-01-30 06:10 下午

U3's new Feature:
1.采用单独的form进行file提交,并将其target设置为隐藏的IFrame,从而实现IFrameIO;
2.集成了先进的ClamAV查毒引擎技术,在文件上传完成前进行病毒扫描,及时删除染毒文件;
3.重构了UploadSessionManager、FileUploadStatusBeanManager、UploadSessionImpl、FileUploadStatus等类,去掉了FileUploadStatus类中的URLList属性,增加了FileUploadStatusList类对其进行FileUploadStatus列表进行收纳、管理。
4.将原来页面功能合为一页,实现了One Page One Application。

#54 评论作者: xmlspy 发表时间: 2007-01-30 06:32 下午

病毒扫描总是报 文件被感染病毒,
文件被删除,导致在页面选择删除的时候失败

#55 评论作者: cleverpig 发表时间: 2007-01-30 10:30 下午

to xmlsply同学:

这并不是bug。使用ClamAV进行文件查毒,需要下载clamwin.
在安装后确认程序中的clamav.properties配置正确,便可使用查毒功能。如果没有安装的话,程序默认把上传文件作为染毒文件对待。

#56 评论作者: cleverpig 发表时间: 2007-01-30 10:32 下午

如果使用unix系统的话,请下载clamav,并安装和配置clamav.properties。
U3版中的clamav.properties支持clamwin在window平台下的默认安装。

#57 评论作者: cleverpig 发表时间: 2007-01-31 04:15 下午

为了增加查毒功能的可选性,更新了部分代码。
大家可以通过设置clamav.properties来可选地开关查毒功能:


#查毒功能使能标志
enable=true
#执行程序路径
executeFile=C:/Program Files/ClamWin/bin/clamscan.exe
#病毒代码库路径
libPath=C:/Documents and Settings/All Users/.clamwin/db



新的U3_1版代码如下:
U3_1 part1
U3_1 part2

#58 评论作者: blueice 发表时间: 2007-01-31 04:46 下午

我装了U3版,挺好,谢谢分享

#59 评论作者: Tin 发表时间: 2007-02-01 09:46 上午

谢谢Cleverpig,很干净的实现,钻研的精神令人敬佩呀!

#60 评论作者: ndlgyb 发表时间: 2007-02-01 10:46 上午

为什么有时会出现 “对不起,网络通讯失败,请重新刷新!”的警告呢
用两台机器测试在上传接近100M的文件能成功,但删除时会出现找不到路径之类的问题

#61 评论作者: cleverpig 发表时间: 2007-02-01 11:54 上午

to ndlgyb同学:

>>为什么有时会出现 “对不起,网络通讯失败,请重新刷新!”的警告呢
后台可能发生错误,请仔细查看一下日志。

>>用两台机器测试在上传接近100M的文件能成功,但删除时会出现找不到路径之类的问题
两种可能:
1。后台可能发生错误,请仔细查看一下日志。
2。文件正在被别的程序访问或者已经被删除,可通过浏览upload目录进行验证。

#62 评论作者: adam 发表时间: 2007-02-06 04:52 下午

无法下载呀,下载为html样式,我修改.html为.jar后,为如下文件。2007_01_31_160627_uSLgPxqmcU_bak.rar,2007_01_31_160637_UqKVpbIsNN.rar解包出现错误。

请问是什么原因?谢谢

#63 评论作者: adam 发表时间: 2007-02-06 04:52 下午

无法下载呀,下载为html样式,我修改.html为.jar后,为如下文件。2007_01_31_160627_uSLgPxqmcU_bak.rar,2007_01_31_160637_UqKVpbIsNN.rar解包出现错误。

请问是什么原因?谢谢

#64 评论作者: adam 发表时间: 2007-02-06 04:52 下午

无法下载呀,下载为html样式,我修改.html为.jar后,为如下文件。2007_01_31_160627_uSLgPxqmcU_bak.rar,2007_01_31_160637_UqKVpbIsNN.rar解包出现错误。

请问是什么原因?谢谢

#65 评论作者: cleverpig 发表时间: 2007-02-07 07:28 下午

我测试了可以下载的,记住要首先解压缩2007_01_31_160627_uSLgPxqmcU.rar文件,然后选择第二个文件2007_01_31_160637_UqKVpbIsNN.rar。

#66 评论作者: fffff 发表时间: 2007-02-25 04:24 上午

ddddddddddddddddddddddddd

#67 评论作者: xiequnwe 发表时间: 2007-02-28 03:41 下午

不知道下面这个问题算不算一个bug:
当浏览一个文件,不上传,然后点取消,这个时候居然发现后台一直在操作。文件没上传上去,但是文件列表里居然有刚才浏览的文件名。

#68 评论作者: xiequnwe 发表时间: 2007-02-28 03:47 下午

重新测试的时候居然发现没有了。怪。

#69 评论作者: xiequnwe 发表时间: 2007-02-28 04:01 下午

上传过程中,如果取消了操作,最好是给于提示,比如正在删除上传的文件,删除完了后提示删除成功。

#70 评论作者: xiequnwe 发表时间: 2007-02-28 04:03 下午

对于上传文件列表中的文件,直接打开,总是报错,不能查看。

#71 评论作者: xiequnwe 发表时间: 2007-02-28 04:03 下午

对于上传文件列表中的文件,直接打开,总是报错,不能查看。

#72 评论作者: xiequnwen 发表时间: 2007-02-28 05:39 下午

这里有个进度条显示的,没用ajax,希望楼主参考,做个完美版本出来。
http://blog.csdn.net/lxzjsj/archive/2006/12/17/1446082.aspx

#73 评论作者: liukui 发表时间: 2007-03-08 05:15 下午

不错,写得不错jlkjkll

#74 评论作者: 夜羽·隼 发表时间: 2007-03-10 02:38 下午

请问:

我的文件放到eclipse里面的时候,他报错:import junit.framework.TestCase;找不到

还有,我在局域网里其他机子上可以上传下载,但是无法直接打开上传的文件

这是为什么呢?

#75 评论作者: kris_zhang 发表时间: 2007-03-14 11:47 上午

如果想在上传后转到一个成功页面,那就在upload.js中refreshUploadStatus函数中if (uploadInfo.status=='upload_done'){
写上要跳转的方法.

#76 评论作者: kris_zhang 发表时间: 2007-03-14 11:48 上午

如果有人有这个的flash版,请给我一份谢谢。
zhang.kris@gmail.com

#77 评论作者: LuckyKing 发表时间: 2007-04-04 07:37 下午

u3在tomcat6.0上测试通过没有问题,但在resin3.0上就一直在初始化的时候循环,不知道是为什么,用的是jdk5.0, u1,u2都无法在tomcat和resin上跑....,不知道楼主有没有测试过这些情况

#78 评论作者: lizhaosuper 发表时间: 2007-04-09 09:42 下午

确实做的还不错支持一下

#79 评论作者: songhuan 发表时间: 2007-04-11 10:24 上午

为什么我来是出来这些
09:29:13,718 DEBUG Digester:1262 -   New match='web-app/mime-mapping'
09:29:13,718 DEBUG Digester:1273 -   Fire begin() for CallMethodRule[methodName=addMimeMapping, paramCount=2, paramTypes={java.lang.String, java.lang.String}]
09:29:13,718 DEBUG Digester:2701 - Pushing params
09:29:13,718 DEBUG sax:932 - characters(

#80 评论作者: xinghai 发表时间: 2007-05-08 11:02 上午

InteractionMessage 这是哪个包的类啊

#81 评论作者: liuwei3230 发表时间: 2007-05-15 09:52 上午

好文章,现在正在学习这方面的.

#82 评论作者: wq163 发表时间: 2007-06-18 01:19 下午

关于对文件上传状态Bean的序列化/反序列化工作,用JSON来做是不是更简单呢?这样就不用在java和js中写那么多解析的代码了

#83 评论作者: xiao7cn 发表时间: 2007-07-15 09:48 下午

cleverpig , 我如果把fileUpload_single.html文件放进里层目录里,上传时提示:“对不起,网络通讯失败,请重新刷新”。请问如何解决?谢谢!

#84 评论作者: xiao7cn 发表时间: 2007-07-15 09:50 下午

cleverpig , 我如果把fileUpload_single.html文件放进里层目录里,上传时提示:“对不起,网络通讯失败,请重新刷新”。请问如何解决?谢谢!

posted on 2007-08-07 17:02 轩辕 阅读(2482) 评论(1)  编辑  收藏 所属分类: java

评论

# re: 掌控上传进度的AJAX Upload  回复  更多评论   

childNodes
2011-01-12 22:44 | childNodes

只有注册用户登录后才能发表评论。


网站导航: