SpringMvc---文件的上传和下载
使用ResponseEntity实现文件下载
首先在webapp的目录下创建一个目录用来存放要被下载的内容,先随便放进去一个图片
然后,打开maven管理,找到Lifecycle-->>package,双击重新打包。把刚才创建的目录打包进要部署的包中
接下来创建一个负责接收下载请求的标签或者页面,然后就是写控制器方法
今天状态不好,控制器没写注解声明,出了半天错,我还以为咋了
在controller中
上面说到,要把ResponseEntity作为控制器方法的返回值,它使用是一般就是作为文件下载使用
在下面的方法中返回值类型是一个ResponseEntity类型的byte数组
其他的每一句都写了作用了。
@RequestMapping("/testDown")
public ResponseEntity<byte[]> testResponseEntity(HttpSession session) throws IOException {
//获取ServletContext对象
ServletContext servletContext = session.getServletContext();
//获取服务器中文件的真实路径,在webapp下
String realPath = ((ServletContext) servletContext).getRealPath("/static/img/1.jpg");
//创建输入流
InputStream fileDown = new FileInputStream(realPath);
//创建字节数组,is.available就是判断是不是有这个待下载文件
byte[] bytes = new byte[fileDown.available()];
//将流读到字节数组中
fileDown.read(bytes);
//创建HttpHeaders对象设置响应头信息
MultiValueMap<String, String> headers = new HttpHeaders();
//设置要下载方式以及下载文件的名字,这句话只可以更改文件名,其他的不能改。attachment是说文件将以附件的形式下载
headers.add("Content-Disposition", "attachment;filename=1.jpg");
//设置响应状态码,HttpStatus是一个枚举类,里面封装了浏览器的响应状态码。就是希望返回200.OK
HttpStatus statusCode = HttpStatus.OK;
//创建ResponseEntity对象
ResponseEntity<byte[]> responseEntity = new ResponseEntity<>(bytes, headers, statusCode);
//关闭输入流
fileDown.close();
return responseEntity;
}
实现文件上传功能
首先要引入依赖
<!--文件上传的依赖-->
<dependency>
<groupId>commons-fileupload</groupId>
<artifactId>commons-fileupload</artifactId>
<version>1.3.1</version>
</dependency>
然后在SpringMvc的配置文件中进行配置。 SpringMVC中将上传的文件封装到MultipartFile对象中,通过此对象可以获取文件相关信息,所以要在配置文件中注册这个接口的实现类CommonsMultipartResolver。id要有,这里SpringMvc就是根据id找bean,没id就找不到就意味着这个bean没注册,id设置为multipartResolver
再就是检查命名空间的引入mvc、context、有没有开放对静态资源的访问、开没开mvc的注解驱动、开没开组件扫描
<!--
注册CommonsMultipartResolver,让SpringIOC容器自动扫描,并且一定要设置id为multipartResolver,否则IOC容器扫描不到这个注册的bean
-->
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"></bean>
在html中
在上传文件时一般是把本地文件上传到服务器端,现在学过的手段中可以使用form表单提交数据,input控件的file类型
在使用form表单时要注意,首先必须是post请求,然后注意提交的地址,再就是必须要有属性enctype=multipart/form-data,使用input控件时将属性type规定为file,name值就是提交参数,控制器方法要接受的,name值要和控制器方法中的参数名一致,或者用@PathVariable注解声明把这个名字的参数赋值给谁。再就是弄一个提交按扭
<body>
<a th:href="@{/testDown}">文件下载</a><br>
<form th:action="@{/testUp}" method="post" enctype="multipart/form-data">
<input type="file" name="photo"/><br>
<input type="submit" name="上传"/>
</form>
</body>
控制器方法
在开头有说到,SpringMvc把上传的文件封装为MultipartFile对象,所以创建一个Multipart类型的参数接收表单提交过来的文件,所以就要参数名和文件名一样。
在负责上传的方法里,可以在mutlipart对象里面找到上传的文件的文件名
然后要找到文件上传进来后要保存的路径【目录】,获取工程在服务器上的路径的方法学过的里面会用的就是ServletContext对象的getRealPath方法。
在实际开发中添加文件上传功能时,上传的文件的保存目录不一定存在,就需要进行创建。
使用File对象代替获取到的保存文件的路径,再对file对象进行判断,如果不存在就创建一个。
最后确定文件最终保存的路径以及文件名。
调用MultipartFile对象的方法transferTo将文件上传到最终目录。
最后就能在maven实际运行的文件夹【target】的目录中看见这个新创建的文件夹和上传过来的文件
在实际操作中发现,如果文件名相同的话,上一次上传的文件就会被覆盖掉。为了解决这个问题,可以使用一个永远不会重复的内容与文件名进行拼接。在对文件名进行拼接形成一个新的文件名时,需要获取到文件的后缀名,使用substring方法对文件名进行截取,在方法内用lastIndexOf方法告诉程序从最后一个点开始截取。这样就能得到完整的文件后缀,再与你准备的永远不重复的String变量进行拼接字符串
@RequestMapping("/testUp")
public String testUp(MultipartFile photo, HttpSession session) throws IOException {
//获取上传的文件的文件名
String fileName = photo.getOriginalFilename();
//处理文件重名问题
String lastName = fileName.substring(fileName.lastIndexOf("."));
fileName = UUID.randomUUID().toString() + lastName;
//获取服务器中photo目录的路径
ServletContext servletContext = session.getServletContext();
String photoPath = servletContext.getRealPath("photo");
File file = new File(photoPath);
if(!file.exists()){
file.mkdir();
}
String finalPath = photoPath + File.separator + fileName;
//实现上传功能
photo.transferTo(new File(finalPath));
return "success";
}