WebWork文件上传问题解析

开发 后端
本文向您介绍作者在进行WebWork文件上传时遇到的问题和解决方法。WebWork上传文件是两次拷贝过程,首先从request的输入流中将文件流输出到一个临时文件,然后你再将此临时文件拷贝到你需要指定的路径。

今天遇到一个webwork文件上传的问题,上传 txt 文件的时候,如果文件内容为空,则上传后的文件为null,上传不成功。试了一下,如果给txt文件写一个二进制的0x00字符,即可以上传成功,如果是上传一个没有内容的word文档,也可以成功,分析原因应该是word格式自带了很多隐藏格式数据,所以其实没有文字内容的word文档也是有东西的。这样看来,只要文件实际内容为空,即不带任何字符,上传就有问题。

随后,我查看了一下webwork.properties文件,发现我们项目的webwork.multipart.parser选用的是jakarta,

于是我依次换用了另外两种parser:pell和cos。最后发现,只有cos不会出错,能够上传成功。看来,是具体parser对上传文件流的解析不一样。

三种parser都是市面上早已存在的,webwork只不过是对它们进行了封装调用,并不是靠自己来实现的。webwork提供了一个通用的访问接口MultiPartRequest,然后针对三种parser,分别继承实现了CosMultiPartRequest、PellMultiPartRequest、JakartaMultiPartRequest。默认情况下,如果你不在webwork.properties文件中设置parser,webwork会选择pell作为parser。

三种parser的区别是:只有Jakarta能做多文件的同时上传;

只有pell能自动支持中文名文件的上传,其他两种需要你自己手动做encoding;cos功能比较强大,比如我上面提到的txt文件内容为空,cos可以上传成功,但其他两种parser就不行,不过webwork的封装使它丧失了很多功能。

另外,webwork.properties文件里还有其他几个与文件上传相关的参数,比如webwork.multipart.saveDir用于设定上传文件的临时文件保存目录,webwork.multipart.maxSize用于设置上传文件的最大字节数。

我又上网找了一篇文章,深入介绍了WebWork文件上传的机制和过程,还找了一篇剖析webwork源码的pdf《Anatomy Webwork Source Code》,大家可以去下载看看 http://public.iecn.net/Along/Anatomy%20Webwork%20Source%20Code_V0.9.pdf

那篇深入介绍Webwork文件上传机制的文章(http://www.wangchao.net.cn/bbsdetail_267965.html)如下:

点击上传按钮后,webwork的程序流如下:

  1. step1)进入ServletDispatcher.service  
  2. publicvoidservice  
  3. (HttpServletRequestrequest,HttpServletResponseresponse)  
  4. throwsServletException{  
  5. ........  
  6. request=wrapRequest(request);  
  7. .........  
  8. }  
  9. step2)进入ServletDispatcher.wrapRequest  
  10. protectedHttpServletRequestwrapRequest  
  11. (HttpServletRequestrequest)throwsIOException{  
  12. ........................  
  13. if(MultiPartRequest.isMultiPart(request)){  
  14. request=newMultiPartRequestWrapper  
  15. (request,getSaveDir(),getMaxSize());  
  16. }  
  17. returnrequest;  
  18. }  
  19. step3)进入MultiPartRequestWrapper的构造方法  
  20. publicMultiPartRequestWrapper  
  21. (HttpServletRequestrequest,StringsaveDir,intmaxSize)  
  22. throwsIOException{  
  23. .....................  
  24. //step3.1)获取webwork.preperties配置的parser  
  25. Stringparser="";  
  26. parser=Configuration.getString("webwork.multipart.parser");  
  27. //Ifit'snotset,usePell  
  28. if(parser.equals("")){  
  29. log.warn("Propertywebwork.multipart.parsernotset."+  
  30. "Usingcom.opensymphony.webwork.dispatcher.multipart.  
  31. PellMultiPartRequest");  
  32. parser="com.opensymphony.webwork.dispatcher.  
  33. multipart.PellMultiPartRequest";  
  34. }  
  35. //legacysupportforoldstylepropertyvalues  
  36. elseif(parser.equals("pell")){  
  37. parser="com.opensymphony.webwork.dispatcher.  
  38. multipart.PellMultiPartRequest";  
  39. }elseif(parser.equals("cos")){  
  40. parser="com.opensymphony.webwork.dispatcher.  
  41. multipart.CosMultiPartRequest";  
  42. }elseif(parser.equals("jakarta")){  
  43. parser="com.opensymphony.webwork.dispatcher.  
  44. multipart.JakartaMultiPartRequest";  
  45. }  
  46. //step3.2)获取后通过反射实例化parser  
  47. try{  
  48. ClassbaseClazz=com.opensymphony.webwork.dispatcher.  
  49. multipart.MultiPartRequest.class;  
  50. Classclazz=Class.forName(parser);  
  51. //makesureitextendsMultiPartRequest  
  52. if(!baseClazz.isAssignableFrom(clazz)){  
  53. addError("Class'"+parser+"'doesnotextendMultiPartRequest");  
  54. return;  
  55. }  
  56. //gettheconstructor  
  57. Constructorctor=clazz.getDeclaredConstructor(newClass[]{  
  58. Class.forName("javax.servlet.http.HttpServletRequest"),  
  59. java.lang.String.class,int.class  
  60. });  
  61. //buildtheparameterlist  
  62. Object[]parms=newObject[]{  
  63. request,saveDir,newInteger(maxSize)  
  64. };  
  65. //instantiateit  
  66. multi=(MultiPartRequest)ctor.newInstance(parms);  
  67. .................................................  
  68. }  

//step4 进入JakartaMultiPartRequest的构造方法(我在webwork配置的parser是Jakarta所以进入了这个方法,如果你配置不同的parser回进入不同的parser

  1. public JakartaMultiPartRequest  
  2. (HttpServletRequest servletRequest,   
  3. String saveDir, int maxSize)  
  4. throws IOException {  
  5. //设置保存参数  
  6. DiskFileUpload upload = new DiskFileUpload();  
  7. // we must store all uploads on disk because   
  8. the ww multipart API is missing streaming  
  9. // capabilities  
  10. upload.setSizeThreshold(0);  
  11. upload.setSizeMax(maxSize);  
  12. if (saveDir != null) {  
  13. upload.setRepositoryPath(saveDir);  
  14. }  
  15. // Parse the request  
  16. try {  

//此方法生成文件,将请求中的每个参数都生成一个临时文件比如upload_00000017.tmp, upload_00000018.tmp等,就算是form提交的参数也如此

  1. List items = upload.parseRequest(servletRequest);  
  2. ......................  
  3. }  

执行完第四步,然后推出ServletDispatcher.wrapRequest,进入serviceAction方法,开始action及其拦截器的栈调用

进入action和调用栈后,拦截器或action可通过如下代码访问上传的临时文件

  1. MultiPartRequestWrapper wrapper =   
  2. (MultiPartRequestWrapper) req;  
  3. File doc = wrapper.getFiles("doc")[0];  

从上面的分析可以看出:

1)WebWork文件上传在进入action栈之前不修改源码或者做一些扩展、覆盖之类的动作,在进入action栈的时候文件已经上传,而且其文件名很难跟踪(upload_00000017.tmp,到底是00000017,0000018,或者0000022等等),毕竟有很多人上传文件,所以临时文件名很难确定,所以如果你想知道上传的进度很难。


2)利用WebWork文件上传是两次拷贝过程,webwork首先从request的输入流中将文件流输出到一个临时文件,然后你再将此临时文件拷贝到你需要指定的路径。

【编辑推荐】

  1. WebWork框架原理与应用(1)
  2. 用WebWork注册页面建立过程
  3. WebWork注入Servlet方法详解
  4. WebWork中返回INPUT的原因
  5. WebWork如何实现文件上传配置过程
责任编辑:冰荷 来源: blog
相关推荐

2009-07-20 14:04:27

WebWork标签嵌套

2010-05-20 13:58:34

2009-07-14 17:20:31

Webwork文件上传

2009-07-08 09:29:58

WebWork

2009-07-14 15:52:00

WebWork文件下载

2009-09-04 10:55:34

2009-07-09 15:55:18

WebWork配置文件

2012-07-24 17:10:10

Linux操作系统

2010-04-29 12:42:09

Unix系统

2010-11-23 15:50:44

MySQL中文建表

2010-02-23 17:57:58

Python部署

2009-11-05 15:36:58

WCF service

2012-05-23 13:11:10

架构存储

2010-01-28 13:15:43

C++参数

2009-07-20 13:29:13

xwork.xmlWebWork

2010-05-24 14:04:48

JavaSwing多线程

2009-12-31 15:08:22

Silverlight

2010-02-22 17:29:47

WCF跨域

2010-04-19 13:54:43

Unix操作系统

2009-09-03 15:33:13

RHEL红帽linux
点赞
收藏

51CTO技术栈公众号