使用HttpURlConnection 发送POST请求上传文件(带参数)

前言

最近在做一个博客的小项目,需要用到文件上传,HttpClient又被Android给弃用了,图片框架暂时还没学。只能使用HttpURLConnection来上传。折腾了好久,今天终于顺利地跟后台完成了对接。因此,写这篇博客梳理一下知识。

理论知识

背景

最早的HTTP POST是 不支持 文件上传的,给编程开发带来很多问题。但是在1995年,ietf出台了rfc1867,也就是《RFC 1867 -Form-based File Upload in HTML》,用以支持文件上传。所以Content-Type的类型扩充了multipart/form-data用以支持向服务器发送二进制数据。因此发送post请求时候,表单属性enctype共有二个值可选,这个属性管理的是表单的MIME编码:

①application/x-www-form-urlencoded( 注:不设置enctype属性时默认为①)
②multipart/form-data


POST的报文请求分析

使用浏览器进行post请求将会发送以下数据:
 

//我是请求头
POST /t2/upload.do HTTP/1.1
Accept-Charset: GBK,utf-8;
Connection: keep-alive
Content-Length: 60408
Content-Type:multipart/form-data; boundary=ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC //设置内容类型为表单类型,同时定义了boundary “界限标识”
Host: w.sohu.com

 //这里开始请求体的地盘啦,第一条请求体的实体数据(字符串参数)
--ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC       //这里是"--"+boundary
Content-Disposition: form-data;name="xxx"   //name="xxx", xxx为要发送的参数名
Content-Type: text/plain; charset=UTF-8     //设置内容类型为text 编码格式为utf-8
Content-Transfer-Encoding: 8bit
 //这里是一个空行(不可少)
116.361545        // 我勒个去(到这里[空行之后的一行]才能写上xxx的参数值)有点坑是吧,我也觉得
--ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC   //这一大串还是"--"+boundary
//第二条请求体的实体数据(图片文件上传)
Content-Disposition: form-data;name="pic"; filename="photo.jpg" //指定了文件
Content-Type: application/octet-stream          //设置了内容类型为application/octet-stream  
Content-Transfer-Encoding: binary
 //还是一个空行(不可少)
[这里是图片二进制数据]
--ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC--

Boundary说明

根据RFC 1867定义,我们需要选择一段数据作为作为请求参数之间的“界限标识” (即boundary属性),这个“边界数据”不能在内容其他地方出现,一般来说使用一段从概率上说“几乎不可能”的数据即可。 
不同浏览器的实现不同 
火狐某次post的 boundary=---------------------------32404670520626, 
operade某次post的 boundary=----------E4SgDZXhJMgNE8jpwNdOAX
例如参数1和参数2之间需要有一个明确的界限,这样服务器才能正确的解析到参数1和参数2。但是分隔符并不仅仅是boundary,而是下面这样的格式:–+ boundary。 
如:boundary为ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC,那么参数分隔符则为: 
--ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC 
不管boundary本身有没有这个”--“(前缀),这个前缀都是不能省略的。 
最后--ZnGpDtePMx0KrHh_G0X99Yef9r8JZsRJSXC--为结束标识

注:以上内容整理自

Multipart/form-data POST文件上传详解
HTTP POST请求报文格式分析与Java实现文件上传


\r (回车) 与 \n (换行)

‘\r’ 回车,回到当前行的行首,而不会换到下一行,如果接着输出的话,本行以前的内容会被逐一覆盖;
‘\n’ 换行,换到当前位置的下一行,而不会回到行首; 
所以在写完每一行数据之后要使用 \r\n才能达到切换至下一行行首的效果

实例

private static final int TIME_OUT = 8 * 1000;                          //超时时间
    private static final String CHARSET = "utf-8";                         //编码格式
    private static final String PREFIX = "--";                            //前缀
    private static final String BOUNDARY = UUID.randomUUID().toString();  //边界标识 随机生成
    private static final String CONTENT_TYPE = "multipart/form-data";     //内容类型
    private static final String LINE_END = "\r\n";                        //换行
/**
     * post请求方法
     * */
    public static void postRequest(final Map<String, String> strParams, final Map<String, File> fileParams) {
        new Thread(new Runnable() {
            @Override
            public void run() {
                HttpURLConnection conn = null;
                try {
                    URL url = new URL(requestUrl);
                    conn = (HttpURLConnection) url.openConnection();
                    conn.setRequestMethod("POST");
                    conn.setReadTimeout(TIME_OUT);
                    conn.setConnectTimeout(TIME_OUT);
                    conn.setDoOutput(true);
                    conn.setDoInput(true);
                    conn.setUseCaches(false);//Post 请求不能使用缓存   
                    //设置请求头参数
                    conn.setRequestProperty("Connection", "Keep-Alive");
                    conn.setRequestProperty("Charset", "UTF-8");
                    conn.setRequestProperty("Content-Type", CONTENT_TYPE+";boundary=" + BOUNDARY);
                    /**
                     * 请求体
                     */
                    //上传参数
                    DataOutputStream dos = new DataOutputStream(conn.getOutputStream());
                    //getStrParams()为一个
                    dos.writeBytes( getStrParams(strParams).toString() );
                    dos.flush();

                    //文件上传
                    StringBuilder fileSb = new StringBuilder();
                    for (Map.Entry<String, File> fileEntry: fileParams.entrySet()){
                        fileSb.append(PREFIX)
                                .append(BOUNDARY)
                                .append(LINE_END)
                                /**
                                 * 这里重点注意: name里面的值为服务端需要的key 只有这个key 才可以得到对应的文件
                                 * filename是文件的名字,包含后缀名的 比如:abc.png
                                 */
                                .append("Content-Disposition: form-data; name=\"file\"; filename=\""
                                        + fileEntry.getKey() + "\"" + LINE_END)
                                .append("Content-Type: image/jpg" + LINE_END) //此处的ContentType不同于 请求头 中Content-Type
                                .append("Content-Transfer-Encoding: 8bit" + LINE_END)
                                .append(LINE_END);// 参数头设置完以后需要两个换行,然后才是参数内容
                        dos.writeBytes(fileSb.toString());
                        dos.flush();
                        InputStream is = new FileInputStream(fileEntry.getValue());
                        byte[] buffer = new byte[1024];
                        int len = 0;
                        while ((len = is.read(buffer)) != -1){
                            dos.write(buffer,0,len);
                        }
                        is.close();
                        dos.writeBytes(LINE_END);
                    }
                    //请求结束标志
                    dos.writeBytes(PREFIX + BOUNDARY + PREFIX + LINE_END);
                    dos.flush();
                    dos.close();
                    Log.e(TAG, "postResponseCode() = "+conn.getResponseCode() );
                    //读取服务器返回信息
                    if (conn.getResponseCode() == 200) {
                        InputStream in = conn.getInputStream();
                        BufferedReader reader = new BufferedReader(new InputStreamReader(in));
                        String line = null;
                        StringBuilder response = new StringBuilder();
                        while ((line = reader.readLine()) != null) {
                            response.append(line);
                        }
                        Log.e(TAG, "run: " + response);
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }finally {
                    if (conn!=null){
                        conn.disconnect();
                    }
                }
            }
        }).start();
    }

    /**
     * 对post参数进行编码处理
     * */
    private static StringBuilder getStrParams(Map<String,String> strParams){
        StringBuilder strSb = new StringBuilder();
        for (Map.Entry<String, String> entry : strParams.entrySet() ){
            strSb.append(PREFIX)
                    .append(BOUNDARY)
                    .append(LINE_END)
                    .append("Content-Disposition: form-data; name=\"" + entry.getKey() + "\"" + LINE_END)
                    .append("Content-Type: text/plain; charset=" + CHARSET + LINE_END)
                    .append("Content-Transfer-Encoding: 8bit" + LINE_END)
                    .append(LINE_END)// 参数头设置完以后需要两个换行,然后才是参数内容
                    .append(entry.getValue())
                    .append(LINE_END);
        }
        return strSb;
    }

 

  • 6
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 发送Post请求是一种用于向服务器发送数据的HTTP请求方法。它常用于提交表单数据、上传文件或者更新服务器上的资源。 要发送Post请求,我们需要以下步骤: 1. 创建一个URL对象,指定请求的URL地址。 2. 创建一个URLConnection对象,用于建立和服务器的连接。 3. 设置URLConnection请求方法为Post,并且设置URLConnection可进行输入和输出。 4. 构造请求参数,并将其写入URLConnection的输出流中。 5. 获取服务器响应,读取URLConnection的输入流。 以下是一个示例代码: ```java import java.io.*; import java.net.*; public class PostRequestExample { public static void main(String[] args) { try { // 创建URL对象 URL url = new URL("http://example.com"); // 创建URLConnection对象 HttpURLConnection connection = (HttpURLConnection) url.openConnection(); // 设置请求方法为Post,并允许输入输出 connection.setRequestMethod("POST"); connection.setDoInput(true); connection.setDoOutput(true); // 构造请求参数 String param = "username=test&password=123456"; byte[] requestData = param.getBytes(); // 写入请求参数 OutputStream outputStream = connection.getOutputStream(); outputStream.write(requestData); outputStream.flush(); outputStream.close(); // 获取服务器响应 InputStream inputStream = connection.getInputStream(); BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream)); String line; StringBuilder response = new StringBuilder(); while ((line = reader.readLine()) != null) { response.append(line); } reader.close(); // 输出服务器响应 System.out.println(response.toString()); // 关闭连接 connection.disconnect(); } catch (MalformedURLException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } ``` 以上是使用Java代码发送Post请求的示例。通过构造合适的请求参数,并将其写入到输出流中,我们可以向服务器发送Post请求,并获取服务器的响应。 ### 回答2: 发送POST请求是一种常见的网络请求方式,它可以向服务器提交数据,并获取服务器返回的响应结果。在发送POST请求时,需遵循以下步骤: 1. 创建一个URL对象,该对象包含请求的URL地址。 2. 打开一个连接到服务器的连接,可以通过URLConnection类的openConnection()方法创建,也可以使用HttpClient等其他网络请求库。 3. 设置请求的方法为POST请求,可以通过URLConnection类的setRequestMethod()方法将请求方法设置为POST。 4. 设置请求头信息,使用setRequestProperty()方法设置请求的头部字段,如Content-Type、Authorization等。 5. 获取输出流,将需要发送的数据写入输出流中,可以通过调用URLConnection类的getOutputStream()方法获取输出流。 6. 发送数据,将写入输出流中的数据发送给服务器。 7. 获取服务器响应,可以通过调用URLConnection类的getInputStream()方法获取响应的输入流。 8. 读取响应数据,从输入流中读取服务器返回的响应数据。 9. 关闭连接和输入输出流,使用close()方法关闭打开的连接以及相关的输入输出流。 总结来说,发送POST请求的流程是创建URL对象,打开连接,设置请求方法和头部信息,获取输出流并写入需要发送的数据,发送数据,获取服务器的响应输入流,读取响应数据,关闭连接和输入输出流。这样就可以成功地发送POST请求并获取到服务器的响应结果。 ### 回答3: 发送POST请求是一种在网络通信中常见的方法。通过POST请求,可以向服务器发送数据,并且能够保留请求的主体部分,这使得POST请求比GET请求更适合发送大量数据或敏感信息。 要发送POST请求,首先需要确定请求的目标URL,即要发送数据的服务器地址。然后,需要创建一个HTTP连接,并指定使用POST方法进行通信。接下来,需要设置请求头,包括Content-Type字段,用于指定请求主体的数据类型。 发送POST请求时,需要将要发送的数据组织成适当的格式,并将其放置在请求主体中。常见的数据格式包括JSON、XML或表单数据。如果是表单数据,可以使用表单编码格式,将数据以key-value的形式进行传输。 最后,使用HTTP连接发送POST请求,并等待服务器的响应。服务器接收到POST请求后,会解析请求主体中的数据,并根据业务逻辑进行处理。服务器处理完请求后,会返回响应数据,可以根据响应的状态码和数据来判断请求是否成功。 总结来说,发送POST请求需要确定目标URL、创建HTTP连接、设置请求头、组织数据格式并放置在请求主体中,然后发送请求并等待响应。这样,我们就可以通过POST请求与服务器进行数据交互。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值