• 详解Ext + Struts2 文件上传(转)


    详解Ext + Struts2 文件上传

    作者:战斗机 http://zpball.iteye.com/blog/1534503

    other : http://www.360doc.com/content/11/1228/23/2812544_175707427.shtml

    前阵子项目里面需要实现文件上传的功能,前后换了包括我在内的三个人来搞,搞了很长时间才搞好,因为一些小问题耽误了很长时间。趁着手热,写下来贴出来给大家分享,希望查到本文的人可以少走一些弯路。 
        项目中用到的技术是ExtJs3.3和SSH等,与文件上传相关的主要是Ext和Struts2。 

        最开始是一同事搞的,使用了Ext示例包里的FileUploadField,需要在页面中引用两个文件:/ext/examples/ux/fileuploadfield/css/fileuploadfield.css和/ext/examples/ux/fileuploadfield/FileUploadField.js。 
        废话少说,直接上代码,中间各种曲折的过程都省略。文件上传的弹框是一个Window,里面内嵌了一个FormPanel。 

    Js代码  收藏代码
    1. var fp = new Ext.FormPanel( {  
    2.        renderTo : Ext.getBody(),  
    3.        fileUpload : true,  
    4.        width : 523,  
    5.        frame : true,  
    6.        autoHeight : true,  
    7.        bodyStyle : 'padding: 10px 10px 0 10px;',  
    8.        labelWidth : 50,  
    9.        defaults : {  
    10.            anchor : '95%',  
    11.            allowBlank : false,  
    12.            msgTarget : 'side'  
    13.        },  
    14.          items : [  
    15.                             new Ext.form.FileUploadField( {  
    16.               buttonText: '浏览...',  
    17.               emptyText: '请选择一个xls文件',  
    18.               name : 'xlsFile',  
    19.               width : 500,  
    20.               buttonCfg: {  
    21.                    40,  
    22.                   iconCls: 'upload-icon'  
    23.               }  
    24.            })   
    25.     ],  
    26.         buttons : [ {  
    27.            text : '上传',  
    28.            handler : function() {  
    29.               if (fp.getForm().isValid()) {  
    30.                   fp.getForm().submit( {  
    31.                      method : 'post',  
    32.                      url :'uploadEmployee.action',// 后台处理的action  
    33.                      waitMsg : '操作处理中,请稍等...',  
    34.                       waitTitle:'提示',  
    35.                      success :function(fp,action){  
    36.                          Ext.Msg.alert('Success''The value of success is: "'+action.result.success+'" on the server');  
    37.                          excelWindow.destroy();  
    38.                      },  
    39.                      failure : function(fp, action) {  
    40.                           var msg = action.response.responseText;  
    41.                             var obj = Ext.decode( msg );  
    42.                             Ext.Msg.alert("提示""Sorry,操作失败,原因:" + obj.message);  
    43.                             excelWindow.destroy();  
    44.                      }  
    45.               });  
    46.                             }  
    47.                    }  
    48.          }]  
    49. });  
    50.    
    51. var excelWindow = new Ext.Window( {  
    52.        renderTo : Ext.getBody(),  
    53.        closeAction : "hide",  
    54.        plain : true,  
    55.        width : 540,  
    56.        title : "批量导入员工信息",  
    57.        modal : true,  
    58.        items:[fp]  
    59.     });  
    60. excelWindow.show();  


    然后,在后台处理的action里面,无论你返回什么值都可以,SUCCESS、NONE、甚至自定义的一个字符串都行。注意这里所说的“返回”是指action里面的处理函数自身的返回,我相信你懂的……。只要你从action向前台返回的json数据中,不包含{success:false}这个值,success:true也可以省略不写,Ext都会认为文件上传操作成功,即会执行submit方法中success属性定义的回调函数。 
    struts配置文件中的部分内容如下: 
    Xml代码  收藏代码
    1. <action name="uploadEmployee" class="uploadEmployeeService">  
    2.        <interceptor-ref name="fileUpload">  
    3.               <param name="maximumSize">10485760</param>  
    4.               <param name="allowedTypes">application/vnd.ms-excel</param>  
    5.        </interceptor-ref>  
    6.        <interceptor-ref name="defaultStack"/>  
    7.        <result name="success" type="json">  
    8.               <param name="contentType">text/html</param>  
    9.         </result>  
    10.         <result name="test1">/employee/jsp/test.jsp  
    11.               <param name="contentType">text/html</param>  
    12.         </result>  
    13. </action>  


    注意ContentType要设定为text/html类型,否则浏览器不能正确处理,可能会将服务器端的响应作为文件提示用户进行下载。要想给前台正确响应数据,可以有多种方法(笔者目前发现了三种): 
    1、可以直接指定action返回json格式的数据,如上面这样result name="success"的返回类型所示。如果采用这种方式,最好在相应的Action类里包含三个属性:success,filename和msg。其中success的类型可以是String,也可以是boolean,另外两个属性是String类型的。如果Action的方法返回的时候,success的值是false,就可以表示文件上传失败,然后msg属性就可以指明文件上传失败的原因;filename可以用来保存文件的名字。 
    注意无论哪种方式,contentType的类型一定要设置为text/html,否则浏览器可能会把服务器端的响应当成文件,提示让你下载。 
    2、可以在一个文本文件里包含一个json格式的字符串,用作服务器端对客户端的反应。如上面result name="test1"时的配置所示。其中指定了Action的返回值为”test1”时,向客户端返回test.jsp页面里包含的内容,而这个页面里面可能只写了一句:{success:true},就相当于服务器端向客户端返回了这个json响应。 
    不过这种方法可能会比较死,不如第一种方法灵活方便,尤其是需要考虑文件上传(或上传后的解析)失败的情况,错误信息不太方便返回。 
    3、可以直接在Action里面,使用HttpServletResponse对客户端进行响应。例如专门写一个写客户端响应的函数如下: 

    Java代码  收藏代码
    1. privatevoid sendMsg(String content) throws IOException{  
    2.         HttpServletResponse response = ServletActionContext.getResponse();  
    3.         response.setCharacterEncoding("utf-8");  
    4.         response.setContentType("text/html");  
    5.         response.getWriter().write(content);  
    6.         response.getWriter().flush();  
    7.         response.getWriter().notify();  
    8. }  


    这样,把要返回给客户端的json内容按照类似{success:true}的标准格式传给sendMsg函数即可。笔者建议在此处调用Writer的flush和notify函数,sendMsg函数发送的数据会立即发送到客户端。所以,你的Action的返回值就可以随心所欲了,即使是INPUT或者ERROR也没有关系。跟第一种方法一样灵活,不过还是不如第一种方法方便。 
    是不是很简单?其实第一个同事搞的时候,就差不多搞好了,主要的问题应该就是他在success回调函数里使用了例子程序中的一个自定义函数msg,而没有把msg函数定义的代码引入进来,结果是即使文件上传成功,执行到success回调函数中的时候,也必然会报错,不能正常关闭上传对话框。 
    另外的原因是,项目中有一个判断“重复登录”的全局函数,使用了Ajax请求从服务器得到的response,用到了response.getResponseHeader()函数做判断。文件上传结束之后,这里总是报错,说是response中没有getResponseHeader()这个方法。最开始我搞的时候,没什么经验,没有太在意这个问题,以为这个跟文件上传无关,就把精力全部集中在解决文件上传本身之上了,结果耽误了很多时间。后来仔细查看API文档的时候,在FormPanel的基类、类Ext.form.BasicForm的描述中,发现了这样一句: 
    The response text is retrieved from the document, and a fake XMLHttpRequest object is created containing aresponseText property in order to conform to the requirements of event handlers and callbacks. 
    即Ext使用FormPanel进行的文件上传时,Ajax请求返回的是一个伪响应(fake XMLHttpRequest),不能当做一般的response来使用。顿时茅塞顿开,修改了该全局函数,使得对于文件上传的response,跳过调用getResponseHeader()函数的判断。问题终于搞定! 
    血的教训!原来所有的Javascript文件中,只要有错误语法错误发生,就可能影响整个页面的所有js的执行啊!不过,在这整个过程中,对稍微Ext有了一点深入的认识,不再是刚进公司的时候,拿来个任务,直接参照着原有的代码进行简单增加和修改了。还了解了什么事Ajax,也学会了使用调试工具等等。 
    Ext还没有捂热,本人就听从项目需要,转arcgis了,可能还得学Flex等,学校期末也临近,要学的东西太多了……不废话咯~ 

    后记:有人问,如果项目中每个ajax请求都会执行某方法,其中用到了getResponseHeader(),那么如何跳过文件上传的Ajax请求的判断呢?例如,我们的项目中Ajax请求返回的数据都是JSON格式的,我使用的方法是,判断如果response是JSON数据,就解析里面有没有我们所上传的文件(找名字即可),这样就可以判断这个ajax请求是不是用于文件上传的,然后决定是否跳过: 

    Js代码  收藏代码
    1. var t = response.responseText;  
    2. if(t.charAt(0)=="{"){  
    3.        if( Ext.decode(t).xlsFileFileName ){  
    4.               return;  
    5.        }  
    6. }  
  • 相关阅读:
    js内置数据类型
    vue禁止复制的方式
    阻止element组件中的<el-input/>的粘贴功能
    Vue插件集合
    qs.parse()、qs.stringify()、JSON.stringify() 用法及区别
    es6数组的一些函数方法使用
    文章段落首字母缩进两个字符
    深圳scala-meetup-20180902(3)- Using heterogeneous Monads in for-comprehension with Monad Transformer
    深圳scala-meetup-20180902(2)- Future vs Task and ReaderMonad依赖注入
    深圳scala-meetup-20180902(1)- Monadic 编程风格
  • 原文地址:https://www.cnblogs.com/damonhuang/p/3053914.html
Copyright © 2020-2023  润新知