最近在做一个文件上传的开放接口,用到Content-Type: multipart/form-data这种请求类型,特地做了一些研究和记录。
在最初的 http协议中,并没有上传文件方面的功能。RFC1867为 http协议添加了这个能力。常见的浏览器,如 Microsoft IE, Mozila, Opera, Chrome,Safari等都已经支持。按照此规范将用户指定的文件发送到服务器。服务器端的程序如 java等,可以按照此规范解析出用户发送来的文件。
RFC1867定义的请求格式如下示例:
<
这里的"----ZcyOpenBoundaryEEpIo3GVWKVCPrX8"是规范中定义的boundary。http传输的内容通过boundary进行了分割,以--${boundary}开始,并以${boundary}--结尾。
明白了以上内容,我们再来看如何使用multipart/form-data进行文件上传。以HttpClient为例进行说明,其他工具大同小异。首先想到的就是要配置 http请求头信息中的Content-Type字段,没错,我们来看如何进行设置:
httpPost.addHeader("Content-Type", "multipart/form-data; boundary=----ZcyOpenBoundaryEEpIo3GVWKVCPrX8");
注意,这里multipart/form-data 后面要跟上boundary。当然,我们也可以不进行Content-Type设置,一般工具都会为我们自动生成规范的Content-Type,自动生成过程不在本次讨论范围内,读者可以自行阅读代码。
继续,我们设置了请求头中的boundary以后还要确保与代码片段1中的boundary保持一致,否则服务端无法读取到请求体信息。服务端正常情况下收到的请求是下面的样子:
当然,上图是以Spring框架为例,其他框架或语言亦大同小异。
那么怎么保证请求头中的boundary与代码片段1中的boundary一致呢?一种办法是模拟http请求手写拼接报文:
String
当然以上方式比较原始,容易出错,我们更喜欢用高级语言。下面还是以HttpClient为例:
String result = "";
String boundary ="----ZcyOpenBoundaryEEpIo3GVWKVCPrX8";
try (CloseableHttpClient httpClient = HttpClients.createDefault()){
String fileName = file.getName();
HttpPost httpPost = new HttpPost(url);
//设置请求头
httpPost.setHeader("Content-Type","multipart/form-data; boundary="+boundary);
MultipartEntity multipartEntity = new MultipartEntity(HttpMultipartMode.STRICT, boundary, Charset.defaultCharset());
...省略内容...
httpPost.setEntity(multipartEntity);
// 执行提交
HttpResponse response = httpClient.execute(httpPost);
if (response.getStatusLine().getStatusCode() == 200) {
//响应
HttpEntity responseEntity = response.getEntity();
if (responseEntity != null) {
// 将响应内容转换为字符串
result = EntityUtils.toString(responseEntity, Charset.forName("UTF-8"));
}
}
} catch (IOException e) {
e.printStackTrace();
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("result=" + result);
注意,上述代码中除了设置header头中的boundary外,还要同时设置MultipartEntity对象中的boundary,这样就保持一致啦。
boundary, Charset.defaultCharset());
至此,服务端已经可以获取到期待已久的文件流信息了。