SpringMvc---文件的上传和下载

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";
}

上一篇:第七节:HttpEntity 与 ResponseEntity


下一篇:【SpringMVC 从 0 开始】文件上传和下载