Android的HTTP客户端
原文地址:
[本文由来自Dalvik团队的Jesse Wilson提交。--Tim Bray]
大多数需要网络连接的Android应用都会使用HTTP来传输数据。Android自带了两个HTTP客户端:HttpURLConnection和Apache HTTP Clinent。它们都支持HTTPS,文件上传下载,配置超时,
IPV6及连接池。
Apache HTTP Client
DefaultHttpClient及其相似的AndroidHttpClient都是HttpClient的扩展适用于Web浏览器。它们有丰富的可扩展的API,同时也很稳定没有什么Bug。
但是这里重量级的API让我们难以修改后保证它的兼容性,因为Android团队在Apache HTTP Client中并没有什么开发者。
HttpURLConnection
HttpURLConnection是一个通用的,轻量级的适用于绝大多数应用的HTTP客户端。这个类比较底层,但我们可以很容易的稳步修改它的关键的API。
在Froyo(2.2)之前,HttpURLConnection有几个烦人的Bug,特别是,在一个可读的InputStream中调用close()会
,一个解决方案就是在低android版本中禁用连接池。
private void disableConnectionReuseIfNecessary() { // HTTP connection reuse which was buggy pre-froyo if (Integer.parseInt(Build.VERSION.SDK) < Build.VERSION_CODES.FROYO) { System.setProperty("http.keepAlive", "false"); }}
在Gingerbread(2.3)中,我们添加了隐式的压缩响应。HttpURLConnection将自动将以下的请求头部添加到连接请求中,然后处理相应的请求:
Accept-Encoding:gzip
为了充分利用这一点,你需要配置你的服务器为支持这一请求头的客户端开启压缩响应。如果压缩响应有问题,
你可以参考HttpURLConnection来禁用它。
因为HTTP客户端的Content-Length头部返回的是压缩后的大小,所以使用getContentLength()来获得解压后的数据块的大小是错误的。你应该从响应中读取字节直到InputStream.read()返回-1。
同时我们对在Gingerbread中的HTTPS连接做了一些优化,当HttpsURLConnection尝试连接SNI(Server Name Indication)将允许多个hosts共享一个IP地址。同样支持压缩和Session tickets。如果连接失败,将自动重试(it is automatically retried without these features)。这使得HttpURLConnection在连接新的服务器时更有效,同时不破坏向下兼容。
在Ice Cream Sandwich(4.X)中,我们添加了响应缓存。有了缓存,HTTP请求就会满足下面三种方式:
(1)完全缓存的响应将直接从本地存在中获得。因为无需进行网络连接 ,这样的响应可以马上获得。
(2)条件缓存的响应必须到服务器验证是否过期,客户端发送一个请求如"给我 /foo.png 如果从昨天后发生了改变的话",然后服务器响应返回更新的内容 或者一个304 Not Modified状态。如果内容没有更新,则将不会下载 。
(3)非缓存的响应将从web端获得,这些请求会存储在响应缓存中以务后用。
可以利用反射来在支持的设备中开启HTTP 响应缓存 。下面的代码将在4.0之后的版本的开启响应缓存 。而不会影响之前的版本。
private void enableHttpResponseCache() { try { long httpCacheSize = 10 * 1024 * 1024; // 10 MiB File httpCacheDir = new File(getCacheDir(), "http"); Class.forName("android.net.http.HttpResponseCache") .getMethod("install", File.class, long.class) .invoke(null, httpCacheDir, httpCacheSize); } catch (Exception httpResponseCacheNotAvailable) { }}
同时你需要配置你的服务器来在HTTP响应中设置缓存头部。
哪一个客户端最好?
在Eclair(2.1)和Froyo(2.2)中,Apache HTTP Client(相比HttpURLConnection)没有什么Bug,
对于这些发行版,它是最好的选择。
对于 Gingerbread(2.3)及之后的版本。HttpURLConnection是最好的选择。它的简单轻量最适合android。
压缩,缓存响应以减少网络连接,提升速度,省电,新的应用应该使用HttpURLConnection,它也是我们将着力点。
PS:决定为HttpURLConnection写一点封装,所以看了下官方博客,见没有翻译,觉得还不错,就翻译了下。
考虑到有些同学不能访问原文,原文如下:
Android’s HTTP Clients
[This post is by from the Dalvik team. —Tim Bray]
Most network-connected Android apps will use HTTP to send and receive data. Android includes two HTTP clients: HttpURLConnection and Apache HTTP Client. Both support HTTPS, streaming uploads and downloads, configurable timeouts, IPv6 and connection pooling.
Apache HTTP Client
and its sibling are extensible HTTP clients suitable for web browsers. They have large and flexible APIs. Their implementation is stable and they have few bugs.
But the large size of this API makes it difficult for us to improve it without breaking compatibility. The Android team is not actively working on Apache HTTP Client.
HttpURLConnection
is a general-purpose, lightweight HTTP client suitable for most applications. This class has humblebeginnings, but its focused API has made it easy for us to improve steadily.
Prior to Froyo, HttpURLConnection had some frustrating bugs. In particular, calling close() on a readable InputStream could . Work around this by disabling connection pooling:
private void disableConnectionReuseIfNecessary() { // HTTP connection reuse which was buggy pre-froyo if (Integer.parseInt(Build.VERSION.SDK) < Build.VERSION_CODES.FROYO) { System.setProperty("http.keepAlive", "false"); } }
In Gingerbread, we added transparent response compression. HttpURLConnection will automatically add this header tooutgoing requests, and handle the corresponding response:
Accept-Encoding: gzip
Take advantage of this by configuring your Web server to compress responses for clients that can support it. If response compression is problematic, the shows how to disable it.
Since HTTP’s Content-Length header returns the compressed size, it is an error to use to size buffers for the uncompressed data. Instead, read bytes from the response until returns -1.
We also made several improvements to HTTPS in Gingerbread. attempts to connect with (SNI) which allows multiple HTTPS hosts to share an IP address. It also enables compression and session tickets. Should the connection fail, it is automatically retried without these features. This makes HttpsURLConnection efficient when connecting to up-to-date servers, without breaking compatibility with older ones.
In Ice Cream Sandwich, we are adding a response cache. With the cache installed, HTTP requests will be satisfied in one of three ways:
-
Fully cached responses are served directly from local storage. Because no network connection needs to be made such responses are available immediately.
-
Conditionally cached responses must have their freshness validated by the webserver. The client sends a request like “Give me /foo.png if it changed since yesterday” and the server replies with either the updated content or a304 Not Modified status. If the content is unchanged it will not be downloaded!
-
Uncached responses are served from the web. These responses will get stored in the response cache for later.
Use reflection to enable HTTP response caching on devices that support it. This sample code will turn on the response cache on Ice Cream Sandwich without affecting earlier releases:
private void enableHttpResponseCache() { try { long httpCacheSize = 10 * 1024 * 1024; // 10 MiB File httpCacheDir = new File(getCacheDir(), "http"); Class.forName("android.net.http.HttpResponseCache") .getMethod("install", File.class, long.class) .invoke(null, httpCacheDir, httpCacheSize); } catch (Exception httpResponseCacheNotAvailable) { } }
You should also configure your Web server to set cache headers on its HTTP responses.
Which client is best?
Apache HTTP client has fewer bugs on Eclair and Froyo. It is the best choice for these releases.
For Gingerbread and better, HttpURLConnection is the best choice. Its simple API and small size makes it great fit for Android.Transparent compression and response caching reduce network use, improve speed and save battery. New applications should use ; it is where we will be spending our energy going forward.