在开发中,我们使用的比较多的HTTP请求方式基本上就是GET、POST。其中GET用于从服务器获取数据,POST主要用于向服务器提交一些表单数据,例如文件上传等。而我们在使用HTTP请求时中遇到的比较麻烦的事情就是构造文件上传的HTTP报文格式,这个格式虽说也比较简单,但也比较容易出错。今天我们就一起来学习HTTP POST的报文格式以及通过Java来模拟文件上传的请求。
首先我们来看一个POST的报文请求,然后我们再来详细的分析它。
POST报文格式
- POST /api/feed/ HTTP/1.1
- Accept-Encoding: gzip
- Content-Length: 225873
- Content-Type: multipart/form-data; boundary=OCqxMF6-JxtxoMDHmoG5W5eY9MGRsTBp
- Host: www.myhost.com
- Connection: Keep-Alive
- --OCqxMF6-JxtxoMDHmoG5W5eY9MGRsTBp
- Content-Disposition: form-data; name="lng"
- Content-Type: text/plain; charset=UTF-8
- Content-Transfer-Encoding: 8bit
- 116.361545
- --OCqxMF6-JxtxoMDHmoG5W5eY9MGRsTBp
- Content-Disposition: form-data; name="lat"
- Content-Type: text/plain; charset=UTF-8
- Content-Transfer-Encoding: 8bit
- 39.979006
- --OCqxMF6-JxtxoMDHmoG5W5eY9MGRsTBp
- Content-Disposition: form-data; name="images"; filename="/storage/emulated/0/Camera/jdimage/1xh0e3yyfmpr2e35tdowbavrx.jpg"
- Content-Type: application/octet-stream
- Content-Transfer-Encoding: binary
- 这里是图片的二进制数据
- --OCqxMF6-JxtxoMDHmoG5W5eY9MGRsTBp--
这里我们提交的是经度、纬度和一张图片(图片数据比较长,而且比较杂乱,这里省略掉了)。
格式分析
请求头分析
- POST /api/feed/ HTTP/1.1
- Accept-Encoding: gzip
- Content-Length: 225873
- Content-Type: multipart/form-data; boundary=OCqxMF6-JxtxoMDHmoG5W5eY9MGRsTBp
- Host: www.myhost.com
- Connection: Keep-Alive
- --OCqxMF6-JxtxoMDHmoG5W5eY9MGRsTBp
我们知道HTTP协议采用“请求-应答”模式,当使用普通模式,即非KeepAlive模式时,每个请求/应答客户和服务器都要新建一个连接,完成之后立即断开连接(HTTP协议为无连接的协议);当使用Keep-Alive模式(又称持久连接、连接重用)时,Keep-Alive功能使客户端到服务器端的连接持续有效,当出现对服务器的后续请求时,Keep-Alive功能避免了建立或者重新建立连接。
如上图中,左边的是关闭Keep-Alive的情况,每次请求都需要建立连接,然后关闭连接;右边的则是Keep-Alive,在第一次建立请求之后保持连接,然后后续的就不需要每次都建立、关闭连接了,启用Keep-Alive模式肯定更高效,性能更高,因为避免了建立/释放连接的开销。
http 1.0中默认是关闭的,需要在http头加入"Connection: Keep-Alive",才能启用Keep-Alive;http 1.1中默认启用Keep-Alive,如果加入"Connection: close ",才关闭。目前大部分浏览器都是用http1.1协议,也就是说默认都会发起Keep-Alive的连接请求了,所以是否能完成一个完整的Keep- Alive连接就看服务器设置情况。
请求实体分析
- --OCqxMF6-JxtxoMDHmoG5W5eY9MGRsTBp
- Content-Disposition: form-data; name="lng"
- Content-Type: text/plain; charset=UTF-8
- Content-Transfer-Encoding: 8bit
- 116.361545
- Content-Type: text/plain; charset=UTF-8
- Content-Transfer-Encoding: 8bit
- Content-Type: application/octet-stream
- Content-Transfer-Encoding: binary
模拟文件上传请求
- public static void uploadFile(String fileName) {
- try {
- // 换行符
- final String newLine = " ";
- final String boundaryPrefix = "--";
- // 定义数据分隔线
- String BOUNDARY = "========7d4a6d158c9";
- // 服务器的域名
- URL url = new URL("www.myhost.com");
- HttpURLConnection conn = (HttpURLConnection) url.openConnection();
- // 设置为POST情
- conn.setRequestMethod("POST");
- // 发送POST请求必须设置如下两行
- conn.setDoOutput(true);
- conn.setDoInput(true);
- conn.setUseCaches(false);
- // 设置请求头参数
- conn.setRequestProperty("connection", "Keep-Alive");
- conn.setRequestProperty("Charsert", "UTF-8");
- conn.setRequestProperty("Content-Type", "multipart/form-data; boundary=" + BOUNDARY);
- OutputStream out = new DataOutputStream(conn.getOutputStream());
- // 上传文件
- File file = new File(fileName);
- StringBuilder sb = new StringBuilder();
- sb.append(boundaryPrefix);
- sb.append(BOUNDARY);
- sb.append(newLine);
- // 文件参数,photo参数名可以随意修改
- sb.append("Content-Disposition: form-data;name="photo";filename="" + fileName
- + """ + newLine);
- sb.append("Content-Type:application/octet-stream");
- // 参数头设置完以后需要两个换行,然后才是参数内容
- sb.append(newLine);
- sb.append(newLine);
- // 将参数头的数据写入到输出流中
- out.write(sb.toString().getBytes());
- // 数据输入流,用于读取文件数据
- DataInputStream in = new DataInputStream(new FileInputStream(
- file));
- byte[] bufferOut = new byte[1024];
- int bytes = 0;
- // 每次读1KB数据,并且将文件数据写入到输出流中
- while ((bytes = in.read(bufferOut)) != -1) {
- out.write(bufferOut, 0, bytes);
- }
- // 最后添加换行
- out.write(newLine.getBytes());
- in.close();
- // 定义最后数据分隔线,即--加上BOUNDARY再加上--。
- byte[] end_data = (newLine + boundaryPrefix + BOUNDARY + boundaryPrefix + newLine)
- .getBytes();
- // 写上结尾标识
- out.write(end_data);
- out.flush();
- out.close();
- // 定义BufferedReader输入流来读取URL的响应
- // BufferedReader reader = new BufferedReader(new InputStreamReader(
- // conn.getInputStream()));
- // String line = null;
- // while ((line = reader.readLine()) != null) {
- // System.out.println(line);
- // }
- } catch (Exception e) {
- System.out.println("发送POST请求出现异常!" + e);
- e.printStackTrace();
- }
- }