Struts2框架本身没有文件上传的功能模块,而是利用现在流行的几个文件上传开源框架,如Common-FileUpload和COS等。Struts2利用拦截器将这些文件上传的框架巧妙的集成进来,不能不被称为一个优秀的拿来主义者。由于拦截器的使用,我们使用Struts2实现文件上传变的非常容易,似乎什么也没发生,文件已经上传到服务器了,但如果仅仅是使用这个功能,可能永远不知道Struts2为你做了多少事情。最近有点闲,分析了下下Struts2源码,才弄清楚其中的真相,笔记一下,以免遗忘。
Struts2利用request解析器来集成其他的文件上传框架,Struts2提供了默认的request解析器JakartaMultiPartRequest,在default.properties的struts.multipart.parser参数就是配置request解析器,默认为jakarta也就是JakartaMultiPartRequest。当然,你也可以设置成cos,不过Struts2没有提供适配cos的request解析器,你得自己实现,然后配置到自己的应用。JakartaMultiPartRequest也就是利用Common-FileUpload来对request进行解析,然后保存解析结果。具体工作就是利用Common-FileUpload的ServletFileUpload对request进行解析,得到所有的FileItem的list,然后对这个list进行分析。如果是文件上传表单,则以表单的name属性分组,以表单name属性为key,对应的FileItem列表为Value保存到一个Map中。JakartaMultiPartRequest的工作已经完成,当然,这个Map中保存的所有的FileItem,框架已经将文件保存到一个临时目录里了,这个临时目录你也可以配置,参数为struts.multipart.saveDir,还要配置一个文件上传总大小,参数为struts.multipart.maxSize。
接着就是FileUploadInterceptor的工作了,他的工作也非常简单。首先进行验证,如文件上传大小限制,上传类型限制验证等。然后以表单名来构造三个参数,例如:你的文件上传表单name为upload,则这三个参数为:upload,uploadContentType,uploadFileName。然后保存到ActionContext的PARAMETERS中,当然如果表单很多,而且表单名又不同,他就会产生不同表单名对应的参数。最后系统的ParametersInterceptor拦截器会将ActionContext的getParameters()得到的各个参数映射赋值给Action的各个属性,这就是为什么我们在开发文件上传的时候一定要在Action中提供与表单名相同的File对象(如果一个表单名又多个上传文件,这是可以是File数组或是File的List),文件名+ContentType,和文件名+FileName 这三种参数。
框架为我们所做的就是这些了,剩下的就是文件拷贝了。现在我们来注意下Action的这几个参数,其中File对象(当然也可能是数组或List),这时他们已经指向系统临时目录里的某个文件,或是系统的内存里的数据了。在我们传输数据时,实际上是从这个临时文件或内存将数据库拷贝到上传目录里去的,当执行完Action之后,FileUploadInterceptor会将临时目录里所有临时文件删除,这也是为什么上传文件完成后会有一些日志显示说删除了一些临时文件的原因。
现在实现一个简单的文件上传,首先熟悉下几个参数配置:
1.struts.multipart.parser 配置request解析器,Struts2默认为jakarta 也就是JakartaMultiPartRequest,我们也 可以自己开发自己的解析器。
2.struts.multipart.saveDir 上传文件的临时保存目录,我们知道Common-FileUpload需要提供一个临时目录的。如果没有设置,则是ServletContext.getAttribute("javax.servlet.context.tempdir")对应的目录,对应于Common-FileUpoad中的repository参数。
3.struts.multipart.maxSize 这个是配置文件上传总大小,单位为B,默认大小为2097152b,对应于Common-FileUpload中的sizeMax参数。
在配置FileUploadInterceptor时候也有几个参数:
1.maximumSize 这是设置允许上传的单个文件的大小,如果某个文件大小大于这个大小,则会产生一个错误保存到Action中,错误的i18n的key为struts.messages.error.file.too.large
2.allowedTypesSet 这是设置允许上传文件的类型,多个类型以逗号“,”隔开,如果某个文件的类型没在这个设置之类,则会产生一个错误保存到Action中,错误的i18n的的key为struts.messages.error.content.type.not.allowed
3.allowedExtensionsSet这是设置允许上传文件的后缀,多个类型以逗号“,”隔开,如果某个文件的后缀没在这个设置之类,则会产生一个错误保存到Action中,错误的i18n的的key为struts.messages.error.file.extension.not.allowed
struts.xml配置示例:
<struts> //配置文件保存的临时目录 <constant name="struts.multipart.saveDir" value="C:\Documents and Settings\bond\My Documents\temp"></constant> //配置上传总大小 <constant name="struts.multipart.maxSize" value="1024102400"></constant> <package name="strutsDemo" extends="struts-default"> <action name="Upload" method="upload" class="com.bond.action.LoginAction"> //文件上传的保存目录 <param name="savePath">/upload</param> <result name="success">/success.jsp</result> <result name="error">/failure.jsp</result> <interceptor-ref name="fileUpload"> //单个文件允许的大小 <param name="maxinumSize">10240000000</param> //允许上传的文件类型,可以到%Tomcat_home%/conf下的web.xml文件中找到所有文件类型的类型名 <param name="allowedTypesSet">application/zip,application/pdf,image/gif</param> </interceptor-ref> <interceptor-ref name="defaultStack"></interceptor-ref> </action> </package> </struts>
这里我们配置了fileUpload拦截器,我们还必须吧dufaultStack加上,由于defaultStack中已经包含了fileUpload拦截器,所以fileUpload拦截器会执行俩遍,所以我们会在日志信息中看到俩次清除临时文件的记录。
UploadAction:
public class UploadAction extends ActionSupport { //必须的参数,参数名与表单名相同,如果一个表单名对应多个上传文件, //则必须为数组或List private File[] upload; //必须的参数,格式:表单名+ContentType,表示上传文件类型 private String[] uploadContentType; //必须的参数,格式:表单名+FileName,表示上传的文件名 private String[] uploadFileName; //文件保存目录 private String savePath; public LoginAction() { } public String upload() throws Exception { for (int i=0;i<getUpload().length;i++) { FileInputStream in = new FileInputStream(getUpload()[i]); FileOutputStream out = new FileOutputStream(getSavePath()+"\\"+getUploadFileName()[i]); //将数据拷贝到上传目录,这里没有必要使用缓冲流,因为IOUtils工具类已经使用了缓存 IOUtils.copy(in, out); } return SUCCESS; } 。。。。。。get/set方法。。。。
页面表单示例:
这样,一个简单的文件上传就实现了