文件上传

Spring 文件上传

我们可以定义如下的模板:

<!DOCTYPE html>
<html>
  <head lang="en">
    <meta charset="UTF-8" />
    <title>文件上传页面</title>
  </head>
  <body>
    <h1>文件上传页面</h1>
    <form method="post" action="/upload" enctype="multipart/form-data">
      选择要上传的文件:<input type="file" name="file" /><br />
      <hr />
      <input type="submit" value="提交" />
    </form>
  </body>
</html>

然后创建文件上传的处理控制器,命名为 UploadController:

@Controller
@Slf4j
public class UploadController {

    @Value("${file.upload.path}")
    private String path;

    @GetMapping("/")
    public String uploadPage() {
        return "upload";
    }

    @PostMapping("/upload")
    @ResponseBody
    public String create(@RequestPart MultipartFile file) throws IOException {
        String fileName = file.getOriginalFilename();
        String filePath = path + fileName;

        File dest = new File(filePath);
        Files.copy(file.getInputStream(), dest.toPath());
        return "Upload file success : " + dest.getAbsolutePath();
    }

}

其中包含这几个重要元素:

  • 成员变量 path,通过@Value 注入配置文件中的 file.upload.path 属性。这个配置用来定义文件上传后要保存的目录位置。
  • GET 请求,路径/,用于显示 upload.html 这个文件上传页面。
  • POST 请求。路径/upload,用于处理上传的文件,即:保存到 file.upload.path 配置的路径下面。

编辑 application.properties 配置文件:

spring.servlet.multipart.max-file-size=2MB
spring.servlet.multipart.max-request-size=2MB

file.upload.path=/Users/didi/

多文件处理

<!DOCTYPE html>
<html>
  <head lang="en">
    <meta charset="UTF-8" />
    <title>文件上传页面 - didispace.com</title>
  </head>
  <body>
    <h1>文件上传页面</h1>
    <form method="post" action="/upload" enctype="multipart/form-data">
      文件1:<input type="file" name="files" /><br />
      文件2:<input type="file" name="files" /><br />
      <hr />
      <input type="submit" value="提交" />
    </form>
  </body>
</html>

可以看到这里多增加一个 input 文件输入框,同时文件输入框的名称修改为了 files,因为是多个文件,所以用了复数。注意:这几个输入框的 name 是一样的,这样才能在后端处理文件的时候组织到一个数组中。

@PostMapping("/upload")
@ResponseBody
public String create(@RequestPart MultipartFile[] files) throws IOException {
    StringBuffer message = new StringBuffer();

    for (MultipartFile file : files) {
        String fileName = file.getOriginalFilename();
        String filePath = path + fileName;

        File dest = new File(filePath);
        Files.copy(file.getInputStream(), dest.toPath());
        message.append("Upload file success : " + dest.getAbsolutePath()).append("<br>");
    }
    return message.toString();
}
  • MultipartFile 使用数组,参数名称 files 对应 html 页面中 input 的 name,一定要对应。
  • 后续处理文件的主体(for 循环内)跟之前的一样,就是对 MultipartFile 数组通过循环遍历的方式对每个文件进行存储,然后拼接结果返回信息。

单元测试

普通接口的单元测试我们是如何写的?看看我们入门例子中的单元测试:

@SpringBootTest
public class Chapter11ApplicationTests {

    private MockMvc mvc;

    @Before
    public void setUp() {
        mvc = MockMvcBuilders.standaloneSetup(new HelloController()).build();
    }

    @Test
    public void getHello() throws Exception {
        mvc.perform(MockMvcRequestBuilders.get("/hello").accept(MediaType.APPLICATION_JSON))
                .andExpect(status().isOk())
                .andExpect(content().string(equalTo("Hello World")));
    }

}

这里我们所用到的核心是 MockMvc 工具,通过模拟 http 请求的提交并指定相关的期望返回来完成。对于文件上传接口,本质上还是 http 请求的处理,所以 MockMvc 依然逃不掉,就是上传内容发生了改变,我们只需要去找一下文件上传的模拟对象是哪个,就可以轻松完成这个任务。

@SpringBootTest(classes = Chapter43Application.class)
public class FileTest {

    @Autowired
    protected WebApplicationContext context;
    protected MockMvc mvc;

    @BeforeEach
    public void setUp() {
        mvc = MockMvcBuilders.webAppContextSetup(context).build();
    }

    @Test
    public void uploadFile() throws Exception {
        MockMultipartFile file = new MockMultipartFile(
                "file",
                "hello.txt",
                MediaType.TEXT_PLAIN_VALUE,
                "Hello, World!".getBytes()
        );

        final MvcResult result = mvc.perform(
                MockMvcRequestBuilders
                        .multipart("/upload")
                        .file(file))
                .andDo(print())
                .andExpect(status().isOk())
                .andReturn();
    }

}
上一页