本篇博客将讲解的是Springmvc的文件上传和下载功能。对于上传功能,我们在项目中是经常会用到的,比如用户注册的时候,上传用户头像,这个时候就会使用到上传的功能。而对于下载,使用场景也很常见,比如我们项目中有个使用说明是是pdf版的,会提供给用户进行下载的功能。相对于来说,这两个功能都是很常见,废话不多说,按照惯例,我们先来看一下本篇博客的目录。
目录
一:搭建SpringMvc开发环境
二:实现文件上传的功能
三:将上传文件绑定到具体的对象上
四 : 实现用户下载的功能
五:总结
一:搭建Springmvc开发环境
首先我们在myeclipse中,新建一个web项目,名为:SpringMvcFileUpload.然后导入必要的jar包,这里的jar包主要是Spring的jar包还有另外一个很重要的jar包,名称为:commons-fileupload-1.3.1.jar,这个是apache开发的一个专门用来上传文件的工具包,其中预置了很多的文件上传的api供我们使用,然后在src目录下新建一个包,名为com.wyq.Controller和com.wyq.domain。两个包分别用来存放控制器和领域模型。然后在webRoot目录下新建Folder,取名jsp,主要用来存放我们的jsp页面,接下来配置web.xml文件,在其中配置上我们的Springmv的控制分发器Servlet用于处理客户端请求的链接,然后新建一个Springmvc-config文件,主要是配置Springmvc的bean,这里除了需要配置视图解析器用来解析视图资源和基本的扫描包之外,还需要配置一个很重要的bean,该bean的名字是“multiPartResolver”,主要是用来处理文件的,其中可以配置的属性为上传文件的大小和编码,我们来具体看一下配置文件的代码:
web.xml文件:
<!--Springmvc的控制分发器-->
<servlet> <servlet-name>springDispatcherServlet</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:springmvc-config.xml</param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>springDispatcherServlet</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping>
springmvc-config.xml:
<context:component-scan base-package="com.wyq" />
<!-- 配置视图解析器 -->
<bean id="viewResolver"
class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/page/"></property>
<property name="suffix" value=".jsp"></property>
</bean>
<bean id="multipartResolver"
class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
<!--上传文件的最大大小>
<property name="maxUploadSize" value="17367648787"></property>
<!-- 上传文件的编码 -->
<property name="defaultEncoding" value="UTF-8"></property>
</bean>
二:实现文件上传的功能
2.1:首先我们来新建一个jsp页面,命名为:uploadform.jsp写一个上传的页面form表单,其中一定要注意的是在表单的属性中添加entcypt="multipart/form-data",这表示上传的将会是二进制流的格式,以规定的二进制进行上传,便于服务器处理,使用post请求:
<body> <form action="gotoAction" enctype="multipart/form-data" method="post"> <table> <tr> <td>请选择文件:</td> <td><input type="file" name="file"></td> </tr> <tr> <td>开始上传</td> <td><input type="submit" value="上传"></td> </tr> </table> </form> </body>
2.2:再写一个上传成功的jsp页面,主要是用来当上传成功时候跳转的页面,命名为:success.jsp
<body> <h2>文件上传成功!</h2> </body>
2.3:然后再写一个上传失败的页面,主要是用来当上传失败的时候跳转的页面,命名为:error.jsp
<body> <h2>文件上传失败!请检查文件是否存在</h2> </body>
2.4:写好了jsp页面,接下来就是写我们的控制器了,Springmvc控制器,在我们的com.wyq.Controller中写一个FileUploadController类,然后写上具体的代码,如下所示,注意其中使用MultipartFile来接受form表单传过来的file文件,MultipartFile有几个核心的api,可以供我们使用,比如 boolean isEmpty(),用来判断文件是否为空。void transferTo( File file),把文件写入目标路径地址.
@Controller
public class FileUploadController {
@RequestMapping(value="/{formName}")
public String loginForm(@PathVariable String formName){
return formName;
}
@RequestMapping(value="/gotoAction",method=RequestMethod.POST)
public String upload(@RequestParam("file") MultipartFile file,
HttpServletRequest request){
if (!file.isEmpty()) {
String contextPath = request.getContextPath();//"/SpringMvcFileUpload"
String servletPath = request.getServletPath();//"/gotoAction"
String scheme = request.getScheme();//"http"
String storePath= "E:\\project\\SpringMvcFileUpload\\WebRoot\\images";//存放我们上传的文件路径
String fileName = file.getOriginalFilename();
File filepath = new File(storePath, fileName);
if (!filepath.getParentFile().exists()) {
filepath.getParentFile().mkdirs();//如果目录不存在,创建目录
}
try {
file.transferTo(new File(storePath+File.separator+fileName));//把文件写入目标文件地址
} catch (Exception e) {
e.printStackTrace();
return "error";
}
return "success";//返回到成功页面
}else {
return "error";//返回到失败的页面
}
}
2.5:好了,到此我们就可以实现上传功能了,发布在Tomact容器里,然后就可以进行上传文件了。在浏览中输入:http://localhost:8080/SpringMvcFileUpload/uploadForm具体的页面请求如下,然后浏览具体的文件,点击上传按钮:
选择好了文件,点击上传
到了这一步还没完,我们再往下看,有没有写入目标文件夹,也就是在D:\userUploadFile\Files目录下看是否存在我的上传文件,查看是存在的也就是正确写入了
三:将上传文件绑定到具体的对象上
3.1:首先我们来定义一个实体类User,这个实体类中的headimage属性的主要作用就是用来映射我们上传的文件,可以看到它是MultipartFile类型的:
public class User implements Serializable{//用户实体类
private String userName;
private MultipartFile headimage;//上传文件会自动绑定到该属性
//省略getter和setter方法
}
3.2:然后写我们的 jsp上传页面,这里我们来模拟一个用户注册上传头像的场景,新建一个registerForm.jsp页面,然后写一个form表单,如下:
<body> <h2>用户注册</h2> <form action="register" enctype="multipart/form-data" method="post"> <table> <tr> <td>用户头像:</td> <td><input type="file" name="headimage"></td> </tr> <tr> <td>上传:</td> <td><input type="submit" value="上传"></td> </tr> </table> </form> </body>
3.3:写我们的方法控制器处理注册逻辑的层的代码,注意其中的user对象加入了@ModelAttribute注解,其主要作用是用来映射把上面的form表单的headimage属性自动注入到对象里面,还有Map<String,Object> map主要是为了存放user对象,放在requestScope里面,这样就可用el表达式把其中的值取出来了。
@RequestMapping(value="/register",method=RequestMethod.POST)
public String reg(@ModelAttribute User user,HttpServletRequest request,Map<String,Object> map){
final String wrong="error";
final String good="success";
MultipartFile headimage = user.getHeadimage();
boolean empty = headimage.isEmpty();
if (!empty) {
String realPath = request.getServletContext().getRealPath("/images");
String uploadPath="D:\\userUploadFile\\Files";
String headimageName = headimage.getOriginalFilename();
File imageFile = new File(uploadPath,headimageName);
try {
headimage.transferTo(new File(uploadPath+File.separator+headimageName));
} catch (Exception e) {
e.printStackTrace();
return wrong;
}
map.put("user", user);
return "userInfo";
}else {
return wrong;
}
}
3.4:我们来新建一个jsp页面,取名为userinfo.jsp,其主要作用是显示刚才的我们的文件名:
<body>
用户头像:${requestScope.user.headimage.originalFilename}
</body>
3.5:页面写完了,我们在来模拟测试一下,在客户端Ie浏览器中,输入http://localhost:8080/SpringMvcFileUpload/registerForm:
四:实现用户下载的功能
4.1:首先我们来改造一下刚才的userInfo.jsp页面,主要目的是为了把它变成一个下载的连接,然后用我们就可以写我们的控制器用来处理href的请求来了。其中可以看到我们使用了el表达式把url传递给下一个控制器,控制器可以取出这些值进行处理:
<body>
<h3>文件下载</h3> -->
<a
href="download?filename=${requestScope.user.headimage.originalFilename}">
用户头像:${requestScope.user.headimage.originalFilename}
</a>
</body>
4.2:来看看我们的download控制器代码:注意download方法返回的是ResponseEntity<byte[]> 类型,这个是封装好的返回类型,我们需要传入byte数组、headers、HttpStatus,然后它就会返回具体的下载流,调用客户端去下载资源
@RequestMapping(value="/download",method=RequestMethod.GET) //匹配的是href中的download请求
public ResponseEntity<byte[]> download(HttpServletRequest request,@RequestParam("filename") String filename,
Model model) throws IOException{
String downloadFilePath="D:\\userUploadFile\\Files";//从我们的上传文件夹中去取
File file = new File(downloadFilePath+File.separator+filename);//新建一个文件
HttpHeaders headers = new HttpHeaders();//http头信息
String downloadFileName = new String(filename.getBytes("UTF-8"),"iso-8859-1");//设置编码
headers.setContentDispositionFormData("attachment", downloadFileName);
headers.setContentType(MediaType.APPLICATION_OCTET_STREAM);
//MediaType:互联网媒介类型 contentType:具体请求中的媒体类型信息
return new ResponseEntity<byte[]>(FileUtils.readFileToByteArray(file),headers,HttpStatus.CREATED);
}
4.3:我们来测试一下写的东西是否能准确运行,点击超链接,注意此链接指向的地址:http://localhost:8080/SpringMvcFileUpload/download?filename="myheadimage".jpg,这就表示这个链接会去请求控制器,然后控制器进行处理下载, 这样就完成了文件的下载功能了:
点击超链接: