<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency>
/** * 防止文件大于10M时Tomcat连接重置 * * @return */ @Bean public TomcatServletWebServerFactory tomcatEmbedded() { TomcatServletWebServerFactory tomcat = new TomcatServletWebServerFactory(); tomcat.addConnectorCustomizers((TomcatConnectorCustomizer) connector -> { if ((connector.getProtocolHandler() instanceof AbstractHttp11Protocol<?>)) { ((AbstractHttp11Protocol<?>) connector.getProtocolHandler()).setMaxSwallowSize(-1); } }); return tomcat; }
# 禁用 thymeleaf 缓存 spring.thymeleaf.cache=false # 是否支持批量上传 (默认值 true) spring.servlet.multipart.enabled=true # 上传文件的临时目录 (一般情况下不用特意修改) spring.servlet.multipart.location= # 上传文件最大为 1M (默认值 1M 根据自身业务自行控制即可) spring.servlet.multipart.max-file-size=10MB # 上传请求最大为 10M(默认值10M 根据自身业务自行控制即可) spring.servlet.multipart.max-request-size=10MB # 文件大小阈值,当大于这个阈值时将写入到磁盘,否则存在内存中,(默认值0 一般情况下不用特意修改) spring.servlet.multipart.file-size-threshold=0 # 判断是否要延迟解析文件(相当于懒加载,一般情况下不用特意修改) spring.servlet.multipart.resolve-lazily=false file.upload.path: /file/upload
@PostMapping("/upload") public Map<String, String> upload(@RequestParam MultipartFile file) throws IOException { //创建本地文件 File localFile = new File(path, file.getOriginalFilename()); //把传上来的文件写到本地文件 file.transferTo(localFile); //返回localFile文件路径 Map<String, String> path = new HashMap<>(); path.put("path", localFile.getAbsolutePath()); return path; }
这时候系统将会出现 FileNotFoundException
,日志类似下面这样:
java.io.FileNotFoundException:C:/Users/cheng/AppData/Local/Temp/tomcat.7543349588424487992.9000/work/Tomcat/localhost/ROOT/file/upload/tmp/file/upload/1558332190813.jpg (系统找不到指定的路径。)
这是什么原因呢?可以进入 transferTo
方法
@Override public void transferTo(File dest) throws IOException, IllegalStateException { this.part.write(dest.getPath()); if (dest.isAbsolute() && !dest.exists()) { // Servlet 3.0 Part.write is not guaranteed to support absolute file paths: // may translate the given path to a relative location within a temp dir // (e.g. on Jetty whereas Tomcat and Undertow detect absolute paths). // At least we offloaded the file from memory storage; it'll get deleted // from the temp dir eventually in any case. And for our user's purposes, // we can manually copy it to the requested location as a fallback. FileCopyUtils.copy(this.part.getInputStream(), Files.newOutputStream(dest.toPath())); } }
而后我们再进入 write
方法
@Override public void write(String fileName) throws IOException { File file = new File(fileName); if (!file.isAbsolute()) { file = new File(location, fileName); } try { fileItem.write(file); } catch (Exception e) { throw new IOException(e); } }
这时候我们看到如果 file.isAbsolute()
成立,也就是我们没有使用绝对路径,那么 file = new File(location,fileName);
会在原来的基础上加上location路径.这就是原因所在,可以通过修改绝对路径解决
在代码中控制
@PostMapping("/upload") public Map<String, String> upload(@RequestParam MultipartFile file) throws IOException { //创建本地文件 String classpath = ResourceUtils.getURL("classpath:").getPath(); File localFile = new File(classpath + path, file.getOriginalFilename()); //把传上来的文件写到本地文件 file.transferTo(localFile); //返回localFile文件路径 Map<String, String> path = new HashMap<>(); path.put("path", localFile.getAbsolutePath()); return path; }
通过 ResourceUtils.getURL("classpath:").getPath()
获得项目路径,然后加上设置的相对路径。
网络上还有一种修改 location
值的方式,可以看 这篇博客
但是我个人使用是一直不可以。
或者可以不使用transferTo,代码如下
@PostMapping("/singleFileUpload") public String singleFileUpload(@RequestParam("file") MultipartFile file) throws IOException { byte[] bytes = file.getBytes(); Path filePath = Paths.get(path + file.getOriginalFilename()); Files.write(filePath, bytes); return file.getOriginalFilename(); }
Paths.get
所使用的也是绝对路径,如果您在Windows机器上使用了这种路径(从/开始的路径),那么路径将被解释为相对于当前驱动器,例如
/file/upload/1.txt
而您的项目位于D盘。那么这条路径就会对应这条完整的路径:
D:/file/upload/1.txt
为了简便,以下代码均是使用绝对路径。
@PostMapping("/uploads") public Map<String, String> uploads(@RequestParam MultipartFile[] files) throws IOException { StringBuilder sb = new StringBuilder(); Map<String, String> paths = new HashMap<>(); for (MultipartFile file : files) { //创建本地文件 File localFile = new File(path, file.getOriginalFilename()); //把传上来的文件写到本地文件 file.transferTo(localFile); sb.append(localFile.getAbsolutePath()).append(","); paths.put(file.getOriginalFilename(), localFile.getAbsolutePath()); } //返回localFile文件路径 return paths; }
@PostMapping("/uploadsWithForm") public Map<String, String> uploadsWithForm(@RequestParam String tmpString, @RequestParam MultipartFile[] files) throws IOException { StringBuilder sb = new StringBuilder(); Map<String, String> paths = new HashMap<>(); paths.put("tmpString", tmpString); for (MultipartFile file : files) { //创建本地文件 File localFile = new File(path, file.getOriginalFilename()); //把传上来的文件写到本地文件 file.transferTo(localFile); sb.append(localFile.getAbsolutePath()).append(","); paths.put(file.getOriginalFilename(), localFile.getAbsolutePath()); } //返回localFile文件路径 return paths; }
/** * 测试多文件上传+json接口 * * @param files * @return */ @ApiOperation("测试多文件上传接口") @ApiImplicitParams({ @ApiImplicitParam(name = "jsonMap", value = "json数据", dataType = "Map"), @ApiImplicitParam(name = "files", value = "文件流对象,接收数组格式", required = true, dataType = "__File"), }) @PostMapping(value = "/uploadsWithJson") public Map<String, String> uploadsWithJson(@RequestPart("files") MultipartFile[] files, @RequestPart("jsonMap") Map<String, Object> jsonMap) throws IOException { StringBuilder sb = new StringBuilder(); Map<String, String> paths = new HashMap<>(); System.out.println(jsonMap); for (MultipartFile file : files) { //创建本地文件 File localFile = new File(path, file.getOriginalFilename()); //把传上来的文件写到本地文件 file.transferTo(localFile); sb.append(localFile.getAbsolutePath()).append(","); paths.put(file.getOriginalFilename(), localFile.getAbsolutePath()); } paths.put("jsonMap", JsonUtils.obj2json(jsonMap)); //返回localFile文件路径 return paths; }
呵呵,不好用对不对。项目抛出了个异常, HttpMediaTypeNotSupportedException
。
WARN o.s.w.s.m.support.DefaultHandlerExceptionResolver - Resolved [org.springframework.web.HttpMediaTypeNotSupportedException: Content type 'application/octet-stream' not supported]
所以我们需要添加个转换器类
@Component public class MultipartJackson2HttpMessageConverter extends AbstractJackson2HttpMessageConverter { /** * Converter for support http request with header Content-Type: multipart/form-data */ public MultipartJackson2HttpMessageConverter(ObjectMapper objectMapper) { super(objectMapper, MediaType.APPLICATION_OCTET_STREAM); } @Override public boolean canWrite(Class<?> clazz, MediaType mediaType) { return false; } @Override public boolean canWrite(Type type, Class<?> clazz, MediaType mediaType) { return false; } @Override protected boolean canWrite(MediaType mediaType) { return false; } }
这样就能够识别了
感觉把springboot文件上传所能遇到的坑全踩了个变,心累。
如果需要项目代码,可以去我的 github 中下载;