1 概述
Spring Boot上传文件,根据官方uploadfile示例修改的,可以打成war放到服务器上(笔者使用的是Tomcat).主要步骤是创建异常类,属性类,接口类与控制器类,最后进行少量修改打包部署到服务器上.
2 环境
- win10
- Tomcat 9.0.30
- IDEA 2019.03
- Spring boot 2.2.2 RELEASE
3 新建工程
选择spring initializer:
改一下包名,打包选项这里可以jar可以war,选jar的话可以在build的时候再生成war.
这里用的是模板引擎Thymeleaf,选择spring web与Thymeleaf.
最后点击finish.
4 新建包
4个包,service,properties,controller,exception.
5 exception
处理两个异常,分别是存储异常与存储文件找不到异常.
5.1 StorageException
package
5.2 StorageFileNotFoundException
package
这个构造函数中的cause是引起这个异常的异常,允许空值,如果是空值则表示这个引起这个异常的异常不存在或者未知.
6 properties
新建StorageProperties.java,设定存储文件的位置,就是location的值,可以使用"../../"这样的值,什么也不加的话会在项目路径下新建文件夹,若有同名的文件夹会被删除再重新创建.
注意一下权限的问题,后面部署到Tomcat上面时可能会因为没有写权限而不能写入文件,要确保文件夹拥有写权限.
package
这里使用@ConfigurationProperties会报红,提示没有@EnableConfigurationProperties:
可以先不管,后面会在Main类中添加@EnableConfigurationProperties(StorageProperties.class).
7 service
先加一个StorageService接口:
package
然后新建一个FileSystemStorageService实现该接口:
package
7.1 init
@Override
使用java.nio.file.Files.createDirectories()创建存储目录,可以建立多级目录.
7.2 deleteAll
@Override
使用工具类FileSystemUtils的方法递归删除文件与文件夹.参数是一个File. 下面是方法源码:
public
首先判断根是否为空,不为空的话判断是否是目录,不是目录的话直接删除,是目录的话,利用listFiles()获取所有文件及文件夹,判断是否为空并进行递归删除.
7.3 load
@Override
Path.resolve(String)返回相对于this的路径,具体来说,等于执行
cd rootLocation
返回pwd的值.
7.4 loadAll
@Override
Files.walk遍历目录,返回一个Stream<Path>,返回的Stream包含打开的一个或多个目录的引用,会在Stream关闭时关闭,第二个参数1表示遍历的最大深度.
然后对这个Stream进行filter过滤,这里是把与rootLocation不相等的Path留下,注意是不相等,就是留下filter()中条件为真的Path,不是把条件为真的Path给"删去".
最后进行map,relativize返回参数相对于调用者的路径,这里是返回Stream中的每个Path相对于rootLocation的路径. 对于relativize,无论什么情况下:
Path
都有
a
为真.
7.5 loadAsResource
@Override
这里的Resource是org.springframework.core.io.Resource,是一个接口,可以通过它访问各种资源,实现类有UrlResource,InputStreamResource等,这里利用Path.toUri()把file转换为Resource后,判断这个源是否存在或者是否可读并返回,否则抛出存储文件找不到异常.
7.6 store
@Override
getOriginalFilename()获取文件原名字,然后通过StringUtils.cleanPath()将其标准化,.处理掉"."与"..",然后判断文件是否为空与是否包含相对路径,没有的话利用Files.copy()进行复制,resolve获取filename相对于rootLocation的值,复制选项是REPLACE_EXISTING. StandardCopyOption有三个可选值: - ATOMIC_MOVE:原子性的移动操作,一般在移动文件或目录时使用. - COPY_ATTRIBUTES:复制属性,可以保留源文件或源目录的属性. - REPLACE_EXISTING:替换已存在的文件.
8 controller
新建FileUploadController.
package
8.1 listUploadedFiles
@GetMapping
@GetMapping是@RequestMapping(method = RequestMethod.GET)的简化写法,将HTTP GET路径映射到特定的处理方法上. 方法的参数是spring MVC中的Model,Model实质上是一个Map,添加的key可以在视图中用${key}获取值,比如,这里添加了"files"作为key,则在视图中可用 ${files}获取值.
MvcUriComponentsBuilder可以为Controller指定uri,fromMethod简单地说就是会调用FileUploadController的serveFile(),参数是path.getFileName().toString(),由于serveFile()返回的是Stream<Path>,利用Stream的collect将其转换成List添加到model中,然后返回uploadForm,表示这是视图的名称,会到resource/templates下寻找.
这里说一下RequestMapping与Model:
8.1.1 RequestMapping
可以用@RequestMapping()来映射URL,可以映射到某个类或某个具体方法.@RequestMapping常用的有以下属性:
- value:请求的URL路径,支持URL模板,正则表达式.
- method:HTTP请求方法,如GET,POST,PUT,DELTE等.
- consumes:允许的媒体类型,如consumes="application/json".对应于HTTP请求的Content-Type.
- produces:相应的媒体类型,如produces="application/json",对于HTTP请求的Accept.
- params:请求参数,如params="action=update". - headers:请求头.
Spring提供了简化的@RequestMapping,提供了新的注解来标识HTTP方法: - @GetMapping - @PostMapping - @PutMapping - ...
所以这里的@GetMapping是简化了的@RequestMapping.
8.1.2 Model
可以向Model添加视图所需要的变量,Model主要有以下方法:
Model
addAttribute()添加一个变量,对于两个参数的,使用name作为变量名称,后面的是值,对于只有一个Object的,变量的名字就是类名字首字母小写后转为的java变量. addAttributes()添加多个变量,如果变量存在则覆盖,其中参数为Collection<?>的方法添加变量名时与addAttribute(Object)的命名规范类似. mergeAttributes()也是添加多个变量,不过变量已存在的话会忽略. containAttributte()判断是否存在变量.
8.2 serveFile
@GetMapping
这里的@GetMapping用来表示显示的用来供下载的文件名,@ResponseBody表示直接返回内容而不是视图名,因为默认返回的是视图名称,@ResponseBody对于String直接返回,否则默认使用Jackson进行序列化.
@PathVariable表示这是@GetMapping中的参数的值,可以省略,默认同名,就是形参的名字与GetMapping中的名字一样,从中取值赋给形参,通过filename加载资源后,作为ResponseEntity的请求体. ResponseEntity从HttpEntity继承而来,ResponseEntity.ok()是一个静态方法,表示构建一个状态为"ok"的ResponseEntity,然后添加请求头.
HttpHeaders
content_disposition表示文件是直接在浏览器打开还是下载,attachment表示是要下载,文件名为file.getFilename().
8.3 handleFileUpload
@PostMapping
@PostMapping()与@GetMapping()类似,只不过方法不是GET而是POST.@RequestParam表示请求参数,里面的是请求参数的名字,使用MultipartFile来处理文件上传. RedirectAttributes是用于重定向使用的,可以附带参数,RedirectAttributes有两种带参的形式:
addAttribute
addAttribute()相当于直接在重定向的地址添加
name
这样的形式,会将参数暴露在重定向的地址上.
而addFlashAttribute()隐藏了参数,只能在重定向的页面中获取参数的值,用到了session,session跳转到页面后就会删除对象. handleFileUpload首先保存文件,然后添加一个保存成功的信息,由于Controller中重定向可以返回以"redirect:"或以"forward:"为前缀的URI,因此返回"redirect:/",重定向到根.
8.4 handleStorageFileNotFound
@ExceptionHandler
@ExceptionHandler()注解会处理Controller层抛出的所有StorageFileNotFoundException类及其子类的异常,ResponseEntity.notFound()相当于返回404标识码.
9 main
package
在原来的基础上添加
@EnableConfigurationProperties
与
@Bean
@EnableConfigurationProperties可以为带有@ConfigurationProperties注解的Bean提供有效的支持,将带有@Configuration注解的类注入为Spring的Bean,在这里是使StorageProperties的@ConfigurationProperties生效,如果没有这一行会报红:
@Bean标注在方法上,等价于spring的xml配置文件的<bean>,注册bean对象. CommandLineRunner接口用于应用初始化后去执行一段代码逻辑,这段代码在整个应用周期只执行一次.
10 application.properties
这里可以设置一些环境配置属性,Spring Boot允许准备多个配置文件,在部署时可以指定那个配置文件覆盖默认的application.properties.这里是有关上传文件的设置:
默认如下:
spring
enabled表示允许上传,file-size-threshold表示上传文件超过一定长度就先写入临时文件,单位MB或KB,location是临时文件存放目录,不设定的话使用web服务器提供的临时目录.max-file-size表示单个文件最大长度,默认1MB,max-request-size为单次HTTP请求上传的最大长度,默认10MB,resolve-lazily表示文件和参数被访问的时候再解析成文件.
在这里只需把max-size调大一点即可.
11 测试
这是在本地进行的测试.直接在IDE上点击运行应用,然后打开浏览器输入:
localhost:
12 打包部署到Tomcat上
Spring Boot通常打成jar包或war包,这里部署到Tomcat上的是打成war包.
12.1 改变打包方式
pom.xml中,<packaing>改成war:
12.2 去除Tomcat依赖
Spring Boot默认自带了一个嵌入式的Tomcat,需要把Tomcat依赖方式改为provided. pom.xml中,在<dependencies>添加:
<dependency>
12.3 修改Main类
修改Main类,让其继承SpringBootServletInitializer,重载configure(),同时main()保持不变.
@SpringBootApplication
12.4 路径问题
这个很重要,设置不当的话就无法访问了,主要就是四个路径: - action:
- @GetMapping
- @PostMapping
- redirect
12.4.1 action
这个是绝对路径,要加上/war项目名.
/war项目名/上传路径名
比如这里war项目名是kr,上传路径名是upload.
12.4.2 @GetMapping
这个是相对路径,相对于当前项目的路径,不用加上/war项目名.
/上传路径名
这里是upload.
12.4.3 @PostMapping
与@GetMapping一样,上传路径名.
/上传路径名
12.4.4 redirect
这个是返回的重定向的路径名,相对路径,与上两个一样,也是上传路径名.
/上传路径名
12.5 设置打包名字
在<build>中添加<finalName>,指定打包出来的war名,注意这个要与上面的war项目名一样,这里设置的是kr.
12.6 Maven打包
运行
mvn
即可打包,对于IDEA,可以在IDEA右侧栏的Maven中,打开Lifecycle,选择package:
12.7 打包完成
打包后的war默认放在target下,名字默认为<artifactId>+<version>.
12.8 上传到服务器
上传的话笔者用的是密钥认证的scp:
id_rsa kr.war username@ip:/usr/local/tomcat/webapps
放到服务器的Tomcat下的webapps目录.
12.9 开启Tomcat
进入到Tomcat目录的bin下:
cd /usr/local/tomcat/bin
./startup.sh
如果正在运行的话就不用启动了,因为会自动检测到webapps目录的变化,把新的war自动解包.
12.10 测试
略,与本地测试类似,不过要注意的是上传的文件夹是在tomcat/bin下,想要修改的话可以修改StorageProperties的location.
13 源码
github
码云
14 参考
1.ConfigurationProperties
2.CommandLineRunner
3.RedirectAttribute