概述
鉴于目前很多人无法理解Volley、Retrofit等网络框架的封装。我想那是因为基础知识的匮乏,千丈高楼平地起,想站的高,基础要扎实。学习本文,默认你已经具备网络基础知识。让我们开始吧!
声明
java中最常用的实现HTTP协议的客户端分别是 HttpUrlConnection、HttpClient. 因Android API2.3以后 废弃掉HttpClient,故本文以HttpUrlConnection为例. 本质是一样的.看完本文后,我建议去深入学习OkHttp(新的HTTP客户端,更加强大,Android4.4内部网络部分已经替换)
Java实例一
Java对于网络的支持异常强大,java.net 包设计架构精美,API丰富易用。实现客户端链接服务器并开始通信非常符合人体工程学,实例
注意:因为Client要访问远程Server 并等待服务器响应,所以会阻塞线程。所以必须把网络请求放在子线程内。(Android主线程阻塞5秒即引发ANR)另外要注意,链接HttpUrlConnection使用完毕建议关掉。
最简单的网络请求仅需要5步
1. 创建链接
2.设置网络请求地址
3.打开与服务器通信通道
4.等待服务器响应
5.拿到响应并处理
看,Java底层帮助我们处理了多少复杂的事情,让网络链接变的异常简单。
通过实例一我们拿到服务器响应“ 200 OK” 我们通过抓包工具(Fiddler / Charles) 来看下这次请求
请求地址 http://www.ruanyifeng.com/blog/
第一行 请求方法 GET 路径 /blog/ 协议 HTTP/1.1
User-Agent :中文名为用户代理,简称 UA,它是一个特殊字符串头,使得服务器能够识别客户使用的操作系统及版本、CPU 类型、浏览器及版本、浏览器渲染引擎、浏览器语言、浏览器插件等。
Host 域名
Connection : Keep-Alive的请求时,它也会在响应头中添加一个同样的字段来使用Keep-Alive。这样一来,客户端和服务器之间的HTTP连接就会被保持,不会断开(超过Keep-Alive规定的时间,意外断电等情况除外) HTTP/1.1 默认所有连接都保持,除非显示的设置该属性为close
Accept-Encoding : 是浏览器发给服务器,声明浏览器支持的编码类型的
第一行 协议HTTP/1.1 返回状态码200 返回信息OK
Date: 当前GMT时间
Last-Modified:文档最后修改时间
Content-Length : 内容长度
Content-Type:text/html 返回内容的格式 具体看 Http Content-Type对照表
以上的字段信息高速我们很多信息,想了解更全更多看这里
由上面可以看到默认Java中帮助我们做了很多事情,除了链接网络,多余的事情我们什么都没做。所以Java网络核心代码仅有这几行。但,这只是示例代码,解决的任务场景很局限。接下来我们在此基础上添加并讲解。
Java实例二
通过实例一我们拿到服务器响应“ 200 OK” 但这个信息是服务器的响应,并非业务上的响应,这就是我们为什么平常开发会有2种状态码,第一种即服务器的状态码。当返回200时,我们拿到服务器返回的内容时(一般是JSON串),再判断是否是0,0一般表示成功,再去取真正接口返回的值。若想拿到服务器发送给客户端的具体内容,我们必须解析字节流,并转成可以看懂的文本。
实现显示服务器传过来的内容,仅需要3步(需要JavaIO知识)
1. 获取服务器返回的输入流(返回内容),我们想办法把字节流转换成字符流,并且为了提高效率使用到带有缓存区的Reader
2.遍历打印,直到最后一行
3.关闭Reader
来直观看下这些HTML内容显示的样子
顺便强烈收藏下这个博客
Android示例三
上面的链接请求不带任何参数,在一般请求中,99.9%,我们会对发送的请求带一些参数值。带参数的请求如何处理呢?我们分别实现GET、POST 带参请求。
首先我们通过之前的知识了解到,POST方式跟GET方式提交数据到服务器的区别:
GET一次提交的数据数据量比较小4K内部其实通过组拼url的方式
POST可以提交比较大的数据form表单的形式 流的方式写到服务器
接下来我的业务是请求一个GET接口,该接口成功后返回GMT时间,请求时必须带上要求的参数。
黄色矩形部分就是现在的url 组拼url 看本文的一定了解规则,不细说。
重点提一下,它和POST的参数一样的Key-Value结构,有了共性就可以做封装。先看下POST如何完成的
整个过程也分为5步,非常的简单.
我们常常在封装网络请求时也会使用Apache提供的API更加强大的,封装更好的HttpClient来完成网络链接。这里我就不细致的讲了,贴一段与上面实现同样登录功能的代码,大家自行对比理解。
我们可以看到相比于上面的HttpUrlConnection的POST请求,代码层次的更加简单。对比一下基本没什么不同,重点可以看到,它有一个HttpPost,并且执行httpClient.execute() 有兴趣可以看下源码实现。
补充
设置请求、响应超时,设置请求Header都是一些“边角”知识,比较多,比较碎,这篇文章不做介绍。