Android中HttpURLConnection对象是怎么生成的

 1 try {
 2     URL mUrl = new URL("https://www.jianshu.com/");
 3     HttpURLConnection http = (HttpURLConnection) mUrl.openConnection();
 4     http.setRequestMethod("GET");
 5     http.setConnectTimeout(1024);
 6     http.connect();
 7     int ret = http.getResponseCode();
 8     Log.d("tag", "JianShu response:" + ret);
 9 } catch (Exception e) {
10     e.printStackTrace();
11 }

上面的代码块是在Android中使用HttpURLConnection方式访问网站的最简单的代码演示。本篇文章要讲的是上述代码的前两行的流程。即如果你有一个网址字符串,Android是如何把它封装成HttpURLConnection对象的。

其实关于HttpURLConnection访问互联网完全是属于Java层的知识,可以说与Android没有任何关系。但由于我本人是在 Android4.4 源代码环境下进行分析的,并且在Android源代码下我可以很方便地去给这些源码添加打印来跟踪其流程,因此这篇文章就叫在Android下的HttpURLConnection对象由来跟踪。

首先来看看URL对象的实例化过程。打开源码:

./libcore/luni/src/main/java/java/net/URL.java

图1 URL对象的实例化

 

URL支持以旧的URL对象来实例化新URL对象,不过我想大多数情况下都是只传一个网址参数来实例化的。

图2 URL对象的实例化

 

对于只传一个网址字符串参数进来实例化的情况,上图带三个参数的URL构造方法大多数代码都不需要去理它。这里值得关心的是上图代码第180行调用的setupStreamHandler()方法。setupStreamHandler()方法里就是实例化一个URLStreamHandler对象而已。setupStreamHandler()方法前半部分也不怎么需要去关心它。只看它后半部分的实现。

图3 setupStreamHandler()后半部分的实现

 

这块代码就很通俗易懂了,就是根据你前面传进来的网址的前缀来决定实例化哪个URLStreamHandler的子类而已。本例中我们传递的是Https类型的网址,故而URL类对象中的成员变量streamHandler在实例化过后所指向的类对象是 HttpsHandler 类对象。

./external/okhttp/android/main/java/com/squareup/okhttp/HttpsHandler.java

接着上图2,streamHandler变量完成赋值后就接下去执行第188行的URL解析了,这部分代码没什么好讲的。只要你传递的网址参数是合法的,就不用理会它,在我们的正常项目中,也不会有人故意传个不正确的网址下去。
至此,URL对象的实例化就已经完成了。Android已经将我们的网址字符串转化成了URL对象了。

 

--------------------------------------------------------------------------------------------------

 

下面我们再看看

mUrl.openConnection();

这个代码在执行的时候其内部都发生了些什么。
首先还是找到URL中关于openConnection()方法的实现。

图4 mUrl.openConnection()

 

顺着前面分析到的streamHandler的引用对象一路跟上去。发现它会调到

./external/okhttp/android/main/java/com/squareup/okhttp/HttpHandler.java

图5 httphandler.openConnection()

 

newOkHttpClient()方法的实现如下图所示。

图6 newOkHttpClient()

 

这里,实例化出一个OkHttpClient类对象后调用它的open()方法,将open()方法执行过后得到的对象作为URLConnection对象返回给APK调用者。

./external/okhttp/src/main/java/com/squareup/okhttp/OkHttpClient.java

图7 OkHttpClient

 

这块代码也够清晰明了了。就本篇例子而言,会去实例化HttpsURLConnectionImpl类对象。

./external/okhttp/src/main/java/com/squareup/okhttp/internal/http/HttpsURLConnectionImpl.java

HttpsURLConnectionImpl.java继承自HttpsURLConnection.java

./libcore/luni/src/main/java/javax/net/ssl/HttpsURLConnection.java

而HttpsURLConnection.java又继承自HttpURLConnection.java

./libcore/luni/src/main/java/java/net/HttpURLConnection.java

至此,我们算是弄明白了Android在拿到用户传入的网址后都干了些什么事,并最终把它封装成HttpURLConnection对象的了。

HttpURLConnection http = (HttpURLConnection) mUrl.openConnection();

换言之,文首贴出的代码中的 http 变量指向的类对象其实是上面流程分析中提到的HttpsURLConnectionImpl类的对象。而假若传入的网址是 http:// 形式的,那么它指向的就是HttpURLConnectionImpl类对象了。

 

posted @ 2018-11-02 10:49  大窟窿  阅读(536)  评论(0编辑  收藏  举报