使用 HTTP 协议访问网络

参考可见:
Android网络请求心路历程
浅谈Android中Http请求与缓存(上)

在 Android 6.0 (API 23) 中,Google 已经移除了 Apache HttpClient 相关的类。推荐使用 HttpUrlConnection。


一、简单使用 HttpURLConnection


1. 获取 HttpURLConnection 的实例。
  • new 出一个** URL 对象**,并传入目标的网络地址:
URL url = new URL("http://www.baidu.com");
  • 再调用 **openConnection() **方法获得:
HttpURLConnection connection = (HttpURLConnection) url.openConnection();
2. 设置 HTTP 请求所使用的方法。
  • GET 表示希望从服务器那里获取数据
  • POST 表示希望提交数据给服务器。
connection.setRequestMethod("GET");
connection.setRequestMethod("POST");
3. 进行一些设置,如设置连接超时、读取超时的毫秒数等。
connection.setConnectTimeout(8000);
connection.setReadTimeout(8000);
4. 调用** getInputStream()** 方法获取到服务器返回的输入流。
InputStream in = connection.getInputStream();
5. 最后调用 **disconnect() **方法将这个 HTTP 连接关闭掉。
connection.disconnect();

接受数据示例:

public class MainActivity extends Activity implements OnClickListener {
    public static final int SHOW_RESPONSE = 0;
    private Button sendRequest;
    private TextView responseText;
    private Handler handler = new Handler() {
    public void handleMessage(Message msg) {
        switch (msg.what) {
            case SHOW_RESPONSE:
                String response = (String) msg.obj;
                // 在这里进行UI操作,将结果显示到界面上
                responseText.setText(response);
            }
        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        sendRequest = (Button) findViewById(R.id.send_request);
        responseText = (TextView) findViewById(R.id.response_text);
        sendRequest.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        if (v.getId() == R.id.send_request) {
            sendRequestWithHttpURLConnection();
        }
    }

    private void sendRequestWithHttpURLConnection() {
        // 开启线程来发起网络请求
        new Thread(new Runnable() {
            @Override
            public void run() {
                HttpURLConnection connection = null;
                try {
                    URL url = new URL("http://www.baidu.com");
                    connection = (HttpURLConnection) url.openConnection();
                    connection.setRequestMethod("GET");
                    // 设置可以从服务器读数据,默认为 true 可不设
                    connection.setDoInput(true);
                    connection.setConnectTimeout(8000);
                    connection.setReadTimeout(8000);
                    if(connection.getResponseCode()==HttpURLConnection.HTTP_OK){
                        InputStream in = connection.getInputStream();
                        // 下面对获取到的输入流进行读取
                        BufferedReader reader = new BufferedReader(new InputStreamReader(in));
                        StringBuilder response = new StringBuilder();
                        String line;
                        while ((line = reader.readLine()) != null) {
                            response.append(line);
                        }
                        in.close();
                        Message message = new Message();
                        message.what = SHOW_RESPONSE;
                        // 将服务器返回的结果存放到Message中
                        message.obj = response.toString();
                        handler.sendMessage(message);
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                } finally {
                    if (connection != null) {
                        connection.disconnect();
                    }
                }
            }
        }).start();
    }
}
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.networktest"
    android:versionCode="1"
    android:versionName="1.0" >
    ......
    <uses-permission android:name="android.permission.INTERNET" />
    ......
</manifest>

提交数据示例:

connection.setRequestMethod("POST");
DataOutputStream out = new DataOutputStream(connection.getOutputStream());
out.writeBytes("username=admin&password=123456");

二、进阶


1. 以字符数组获取响应数据

以上是通过 inputSream 来获取响应数据,但 inputSream 用过后就不能使用了(?),有时为了以后再使用响应数据,或某些更方便的情况下,可以用字符数组的方式来获取响应数据。

        ......
        InputStream inputStream = connection.getInputStream()
        byte[] buffer = new byte[1024];
        ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
        int length;
        try {
            // .read(byte[]) 一次读传入缓存区大小,返回读到的字节数,没读到返回 -1
            while ((length = inputStream.read(buffer)) != -1) {
                // byteArrayOutputStream 扩展 length 大小的容量,并写入 buffer 内容
                byteArrayOutputStream.write(buffer, 0, length);
            }
            inputStream.close();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            // 记得关闭
            byteArrayOutputStream .close();
        }

转为 String:

String result=byteArrayOutputStream.toString();

转为 Bitmap:

Bitmap result=BitmapFactory.decodeByteArray(byteArrayOutputStream.toByteArray(),0,byteArrayOutputStream.size());

2. GET 提交参数

GET 方法也可以提交参数,不过不是像 POST 方法把参数放在请求体里,而是在** url 后面加上参数**,形式为XXXXXXXX?param1=value1&param2=value2。如:
http://www.jianshu.com?username=TTT&password=123456

3. POST 提交

POST 提交是编码放在请求体里。

(1)提交参数

(这种键值对参数一般用 application/x-www-form-urlencoded 编码方式最方便,当然也可以用 multipart/form-data编码方式,见下文。)

如:

  POST /meme.php/home/user/login HTTP/1.1 
  Host: 114.215.86.90 
  Cache-Control: no-cache 
  Postman-Token: bd243d6b-da03-902f-0a2c-8e9377f6f6ed 
  Content-Type: application/x-www-form-urlencoded 

  tel=13637829200&password=123456
  • 要在 header 中设置编码方式 Content-Type: application/x-www-form-urlencoded
    connection.setRequestProperty("Content-Type", "application/x-www-form-urlencoded");
  • 例子,把 Map 中的所有参数组装成 body:
    /**
     * 获得body参数字节数组
     */
    public byte[] getBody() {
        StringBuilder bodyString = new StringBuilder();
        byte[] body = null;
        if (params != null && params.size() > 0) {
            try {
                // 把参数拼接为"name=TTT&password=123456"形式的字符串
                for (Map.Entry<String, String> entry : params.entrySet()) {
                    bodyString.append(URLEncoder.encode(entry.getKey(), "UTF-8"))
                            .append("=")
                            .append(URLEncoder.encode(entry.getValue(), "UTF-8"))
                            .append("&");
                }
                // 去掉最后一个“&”
                bodyString.deleteCharAt(bodyString.length() - 1);
                body = bodyString.toString().getBytes("UTF-8");
            } catch (UnsupportedEncodingException e) {
                e.printStackTrace();
            }
        }
        return body;
    }
  • 最后在网络请求中把 body 写入 outputStream。注意要设置可以向服务器写数据:
                // 设置可以向服务器写数据,默认为 false
                connection.setDoOutput(true);
                ......
                outputStream = connection.getOutputStream();
                outputStream.write(body);
(2)上传文件

参考可见:
浅谈Android中Http请求与缓存(上)
Android基于Http协议实现文件上传功能的方法

如:

POST http://58.252.100.248/4.0/blkupload HTTP/1.1
Content-Length: 131950
Content-Type: multipart/form-data; boundary=--------UpsClient.Package
Host: 58.252.100.248
Connection: Keep-Alive
User-Agent: Apache-HttpClient/UNAVAILABLE (java 1.4)
// 空行
----------UpsClient.Package
Content-Disposition: form-data; name="filehash"
// 空行
62661949192F70AA0EE2F21DF99D34B8837E14FD
----------UpsClient.Package
Content-Disposition: form-data; name="bigblkoffset"
// 空行
0
----------UpsClient.Package
Content-Disposition: form-data; name="uploadfile"; filename="file.txt"
Content-Type: application/octet-stream
// 空行
(file.txt 文件的内容)
----------UpsClient.Package--
// 空行
  • 要在 header 中设置编码方式 Content-Type: multipart/form-data; boundary=--------UpsClient.Package
    connection.setRequestProperty("Content-Type", "multipart/form-data")+" ;boundary="+边界标识(String 类型));
  • 边界标识:用来分隔每条数据。可以用UUID.randomUUID().toString();随机生成,也可以自己设定。
    具体使用:--+边界标识+\r\n(换行)
    最后一个要在末尾再加一个--表示结束:--+边界标识+--+\r\n(换行)
  • 对于string类型,只需要 content-disposition 和 name 属性即可。name 是服务器端需要的 key,如 username、password。
    对于文件上传,还需要指定 filename 和 content-type 属性。
  • 注意,例子中的几处空行都不能少。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,905评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,140评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,791评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,483评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,476评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,516评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,905评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,560评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,778评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,557评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,635评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,338评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,925评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,898评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,142评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,818评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,347评论 2 342

推荐阅读更多精彩内容