读书人

spring基于注解的下传

发布时间: 2012-09-06 10:37:01 作者: rapoo

spring基于注解的上传
Spring支持web应用中的分段文件上传。这种支持是由即插即用的MultipartResolver来实现。

在Spring2.0时,Spring提供了两种现成的MultipartResolver
1.Commons FileUpload(http://jakarta.apache.org/commons/fileupload)
2. COS FileUpload(http://www.servlets.com/cos)

不过在spring2.5以后,则只支持Commons FileUpload了。

那么,在spring2.5基于注解的MVC中如何使用MultipartResolver呢,
下面通过一个例子来说明如何使用CommonsMultipartResolver执行附件上传。

首先我们来构建一个web工程,工程结构如下图所示:

关于springMVC的配置这里不再详述,读者可以参看以前的文章。
这里我们只需要在spring-servlet.xml中加入如下内容,用以通知spring加载文件上传处理器:

<!-- 定义文件上传处理器 --> <bean id="multipartResolver" />


当配置了这个处理器后,spring会对用户请求进行拦截,判断是否为附件上传类型,既enctype="multipart/form-data",如果是,则会对请求流进行处理,将其转换为DefaultMultipartHttpServletRequest对象,该对象封装了附件内容,例如:

//取得附件名称列表 Iterator iterator = multipartRequest.getFileNames(); while(iterator.hasNext()){ //循环取出每一个附件 MultipartFile multifile = multipartRequest.getFile((String)iterator.next()); System.out.println(multifile.getOriginalFilename()); }



来看一个例子:
在页面中有两个上传文本框,如下:
 <form action="<%=request.getContextPath() %>/demo/upload.do" enctype="multipart/form-data" method="post"> file1:<input type="file" name="file1" id="file1" /> <br></br> file2:<input type="file" name="file2" id="file2" /> <br></br> <button type="submit">submit</button> </form>

注意,这里一定要将form的enctype属性设置为multipart/form-data,否则不能实现附件上传。

当选择要上传的文件后,点击submit按钮,controller处理方法如下:
@RequestMapping(value = "/demo/upload.do") public String handleImport( @RequestParam(value = "file1", required = false) MultipartFile file1, @RequestParam(value = "file2", required = false) MultipartFile file2, Model model) throws IOException { List<FileModel> list = new ArrayList<FileModel>(); if (file1 != null&&StringUtils.hasText(file1.getOriginalFilename())) { System.out.println(file1.getOriginalFilename()); FileModel fileModel1 = new FileModel(); fileModel1.setName(file1.getOriginalFilename()); fileModel1.setSize(file1.getSize()); String path = service.saveFileToServer(file1, ASVE_PATH); fileModel1.setPath(path); list.add(fileModel1); } if (file2 != null&&StringUtils.hasText(file2.getOriginalFilename())) { System.out.println(file2.getOriginalFilename()); FileModel fileModel2 = new FileModel(); fileModel2.setName(file2.getOriginalFilename()); fileModel2.setSize(file2.getSize()); String path = service.saveFileToServer(file1, ASVE_PATH); fileModel2.setPath(path); list.add(fileModel2); } model.addAttribute("list", list); return "demo/list"; }

说明:
1.@RequestParam(value = "file1", required = false) :将参数中的file1绑定到MultipartFile file1,此时CommonsMultipartResolver已经帮我们把附件内容填充到MultipartFile 中了,这里required = false最好设置为false,除非你确定这个参数一定会传递给controller,否则会抛出参数绑定异常。
2.ASVE_PATH:一个常量,文件保存的路径
3.service.saveFileToServer():该方法将上传文件保存到指定路径。
4.getOriginalFilename():文件名称
5.getSize():文件大小

这里说一下saveFileToServer(),该方法将附件保存到指定路径,并返回保存后的文件全路径:
public String saveFileToServer(MultipartFile multifile, String path) throws IOException { // 创建目录 File dir = new File(path); if (!dir.exists()) { dir.mkdir(); } // 读取文件流并保持在指定路径 InputStream inputStream = multifile.getInputStream(); OutputStream outputStream = new FileOutputStream(path + multifile.getOriginalFilename()); byte[] buffer = multifile.getBytes(); int bytesum = 0; int byteread = 0; while ((byteread = inputStream.read(buffer)) != -1) { bytesum += byteread; outputStream.write(buffer, 0, byteread); outputStream.flush(); } outputStream.close(); inputStream.close(); return path + multifile.getOriginalFilename(); } 

ok,怎么样,spring处理附件上传很简单吧。

这里讨论这样一种情况,如果我实现不知道我要上传多少个附件,也不到对应的参数名称,比如使用FancyUpload这样的多附件上传组件,这又该如何处理呢?

其实很简单,只需要在处理器的方法中加入DefaultMultipartHttpServletRequest multipartRequest参数即可,还拿上面那个页面举例,controller的处理方法如下:
@RequestMapping(value = "/demo/uploadMulti.do") public String handleImport(Model model, DefaultMultipartHttpServletRequest multipartRequest) throws IOException { List<FileModel> list = new ArrayList<FileModel>(); if (multipartRequest != null) { Iterator iterator = multipartRequest.getFileNames(); while (iterator.hasNext()) { MultipartFile multifile = multipartRequest.getFile((String) iterator.next()); if (StringUtils.hasText(multifile.getOriginalFilename())) { System.out.println(multifile.getOriginalFilename()); FileModel fileModel = new FileModel(); fileModel.setName(multifile.getOriginalFilename()); fileModel.setSize(multifile.getSize()); String path = service.saveFileToServer(multifile, ASVE_PATH); fileModel.setPath(path); list.add(fileModel); } } } model.addAttribute("list", list); return "demo/list"; } 

与之前的方法相比,我们不再使用MultipartFile 参数,而改为使用DefaultMultipartHttpServletRequest 参数,其实这样做更为方便,可以批量处理。


另外,我们在配置

CommonsMultipartResolver时可以控制每次上传文件的大小,如下:
<bean id="multipartResolver" value="100000"/> </bean>

PS:如果不想在配置文件中增加配置,又想使用spring对多附件上传的支持,该如何做呢,可以参考下面的例子:
@RequestMapping(value = "/test/upload2.do", method = POST) public String handleImport2(Model model, HttpServletRequest request) { CommonsMultipartResolver commonsMultipartResolver = new CommonsMultipartResolver( request.getSession().getServletContext()); // 设置编码 commonsMultipartResolver.setDefaultEncoding("utf-8"); if (commonsMultipartResolver.isMultipart(request)) { // 转换成多部分request MultipartHttpServletRequest multipartRequest = commonsMultipartResolver.resolveMultipart(request); Iterator iterator = multipartRequest.getFileNames(); while (iterator.hasNext()) { MultipartFile multifile = multipartRequest.getFile((String) iterator.next()); System.out.println(multifile.getOriginalFilename()); } commonsMultipartResolver.cleanupMultipart(multipartRequest); } return null; }

说明:这种方式不能在spring配置文件中声明multipartResolver,否则会得不到上传文件内容

读书人网 >软件架构设计

热点推荐