Volley源码解析

导语

阅读源码是已经入门的Android开发者必经之路,这是提高自己对代码理解的一步。但是一开始阅读源码不能深入的去读细节部分,而是理顺源码的主要部分,理解它的实现原理即可,否则一旦陷入会导致很难读懂,可能会失去阅读源码的兴趣。volley现在已经不是主流的网络请求框架,但是其结构并不复杂,容易理解,因此选择此框架来学习。

一、Volley的使用

RequestQueue requestQueue = 
    Volley.newRequestQueue(context.getApplicationContext());// 创建请求队列实例
StringRequest request = new StringRequest(url,
    new Listener<String>() {
        @Override
        public void onResponse(String response) { // 成功返回数据,做UI更新等操作},
    new ErrorListener() {
         @Override
         public void onErrorResponse(VolleyError arg0) {// 失败,做相应处理}
    });
requestQueue.add(request);// 将请求添加至请求队列中

以上是volley常规的使用方式。先是创建请求队列,这个一般在采用单例模式创建然后用于全局;接着创建请求,请求有StringRequest、JsonObjectRequest等等,主要区别就是返回数据格式不同。最后将请求添加至请求队列(循环处理请求)。

二、Volley解析

阅读源码的方法很简单,就是跟进代码,一点点理解。我们从创建请求队列实例开始跟进。(可能会省略部分不重要的源码)

2.1 Volley

跟进Volley.newRequestQueue(context.getApplicationContext()),可以发现此类是volley框架的入口,源码如下:

    public static RequestQueue newRequestQueue(Context context, HttpStack stack) {
        File cacheDir = new File(context.getCacheDir(), "volley");

        if(stack == null) {// 在api9以前使用httpClient网络请求,而api9以后使用httpUrlConnection,但是在api23以后已经不支持httpClient了
            if(VERSION.SDK_INT >= 9) {
                stack = new HurlStack();// 用httpUrlConnection请求并获取结果
            } else {
                stack = new HttpClientStack(AndroidHttpClient.newInstance(userAgent));// 用httpClient请求并获取结果
            }
        }

        BasicNetwork network1 = new BasicNetwork((HttpStack)stack);// 处理请求,使用httpStack获得请求结果封装后返回
        RequestQueue queue1 = new RequestQueue(new DiskBasedCache(cacheDir), network1);// 创建请求队列,并传入缓存对象以及处理请求对象
        queue1.start();// 启动了缓存线程以及网络请求线程
        return queue1;
    }

2.2 RequestQueue

我们先跟进queue1.start(),读源码的时候不能一直按顺序读,如果先跟进BasicNetWork或DiskBasedCache会很难理解,但是我们能从语义上基本能看出他们的作用,一个是网络请求一个是磁盘缓存。queue1.start()能看出是队列启动的入口,从这里进入会遇到使用BasicNetWork或DiskBasedCache这些类的时候。

    public void start() {
        this.stop();
        this.mCacheDispatcher = new CacheDispatcher(this.mCacheQueue,
             this.mNetworkQueue, this.mCache, this.mDelivery);// 这是一个缓存线程,已缓存结果的一些请求不必再去进行网络请求,直接在mCacheQueue里拿结果,同时缓存会有过期、新鲜度等判断,如果不符合条件则取mNetworkQueue里的网络请求。mCache是缓存结果的一个实例,mDelivery是分发结果的实例。
        this.mCacheDispatcher.start();

        for(int i = 0; i < this.mDispatchers.length; ++i) {
            NetworkDispatcher networkDispatcher = new NetworkDispatcher(this.mNetworkQueue,
                this.mNetwork, this.mCache, this.mDelivery);// 网络请求线程,网络请求时间长,因此需要多个线程一起执行,默认是4个,可以自己设置。
            this.mDispatchers[i] = networkDispatcher;
            networkDispatcher.start();
        }

    }

2.3 NetworkDispatcher

先讲解NetworkDispatcher是因为我想要根据一开始执行的流程走, 只有进行过网络请求的才有结果缓存,这样思路比较顺点。NetworkDispatcher是一个处理网络请求的线程。

public void run() {

        while(true) {
            long startTimeMs;
            Request request;
            while(true) {// 只要网络请求队列里出现请求,马上取出执行。
                startTimeMs = SystemClock.elapsedRealtime();
                try {
                    request = (Request)this.mQueue.take();// 从队列拿出请求
                    break;
                } catch (InterruptedException var6) {
                    if(this.mQuit) {
                        return;
                    }
                }
            }

            try {
                request.addMarker("network-queue-take");
                if(request.isCanceled()) {// 这些地方我们就不必去深究,只要语义上理解就好,很明显这是请求取消了,然后请求完成。传入了一些标志的字符串。
                    request.finish("network-discard-cancelled");
                } else {
                    NetworkResponse e = this.mNetwork.performRequest(request);// 这里就是关键的网络请求,拿到封装好的结果(其实是将httpResponse解析成自定义的NetworkResponse)
                    request.addMarker("network-http-complete");
                    if(e.notModified && request.hasHadResponseDelivered()) {
                        request.finish("not-modified");
                    } else {
                        Response volleyError1 = request.parseNetworkResponse(e);// 再次将NetWorkResponse对象转换成自定义的Response对象。Response最终可解析成我们需要的数据,而NetworkResonse只是过渡的一个对象。
                        request.addMarker("network-parse-complete");
                        if(request.shouldCache() && volleyError1.cacheEntry != null) {
                            this.mCache.put(request.getCacheKey(), 
                                volleyError1.cacheEntry);// 这里就是缓存结果
                            request.addMarker("network-cache-written");
                        }

                        request.markDelivered();
                        this.mDelivery.postResponse(request, volleyError1);// 这也是关键,将结果分发给request,request再传给监听器回调给主线程
                    }
                }
            } catch (VolleyError var7) {
                var7.setNetworkTimeMs(SystemClock.elapsedRealtime() - startTimeMs);
                this.parseAndDeliverNetworkError(request, var7);
            } catch (Exception var8) {
                VolleyError volleyError = new VolleyError(var8);
                volleyError.setNetworkTimeMs(SystemClock.elapsedRealtime() - startTimeMs);
                this.mDelivery.postError(request, volleyError);
            }
        }
    }

2.4 BasicNetwork

该类实现了Network接口,主要的作用是进行网络请求并返回结果。 跟进以上代码的mNetwork.performRequest(request)

    public NetworkResponse performRequest(Request<?> request) throws VolleyError {
        long requestStart = SystemClock.elapsedRealtime();

        while(true) {
            HttpResponse httpResponse = null;
            Object responseContents = null;
            Map responseHeaders = Collections.emptyMap();

            try {
                HashMap e = new HashMap();
                this.addCacheHeaders(e, request.getCacheEntry());// 取出request中的缓存实体,将部分信息缓存到缓存头中(可能无法理解为什么要有缓存头,不用死缠烂打,先看下面)
                httpResponse = this.mHttpStack.performRequest(request, e);// 这里就是真正的请求数据并返回结果。
                StatusLine statusCode2 = httpResponse.getStatusLine();
                int networkResponse1 = statusCode2.getStatusCode();// 状态码,用于判断服务端的结果是否改变
                responseHeaders = convertHeaders(httpResponse.getAllHeaders());// 解析返回结果的响应头
                if(networkResponse1 != 304) {// 304表示在一定时间内结果是否有改动
                    byte[] responseContents1;
                    if(httpResponse.getEntity() != null) {
                        responseContents1 = this.entityToBytes(httpResponse.getEntity());// 将httpEntry实体转换成字节数组
                    } else {
                        responseContents1 = new byte[0];
                    }

                    long requestLifetime1 = SystemClock.elapsedRealtime() - requestStart;
                    this.logSlowRequests(requestLifetime1, request, responseContents1, statusCode2);
                    if(networkResponse1 >= 200 && networkResponse1 <= 299) {
                        return new NetworkResponse(networkResponse1, responseContents1, responseHeaders, false, SystemClock.elapsedRealtime() - requestStart);// 返回一个根据服务端返回结果来构建的NetworkResponse对象
                    }

                    throw new IOException();
                }

                Entry requestLifetime = request.getCacheEntry();// 执行到这里说明服务端没有改动结果,因此直接取缓存
                if(requestLifetime == null) {
                    return new NetworkResponse(304, (byte[])null, responseHeaders, true, SystemClock.elapsedRealtime() - requestStart);
                }

                requestLifetime.responseHeaders.putAll(responseHeaders);
                return new NetworkResponse(304, requestLifetime.data, requestLifetime.responseHeaders, true, SystemClock.elapsedRealtime() - requestStart);
            } catch (SocketTimeoutException var12) {
            }
        }
    }

2.5 HurlStack

很明显,我们要跟进核心部分mHttpStack.performRequest(request, e),这里是通过接口的向上转型,我拿HurlStack实现类来讲解,因为HttpStack已经基本用不到了。

    public HttpResponse performRequest(Request<?> request, Map<String, String> additionalHeaders) throws IOException, AuthFailureError {
        String url = request.getUrl();
        HashMap map = new HashMap();
        map.putAll(request.getHeaders());
        map.putAll(additionalHeaders);

        //这部分是参数的拼接,准备用来请求
        URL parsedUrl1 = new URL(url);
        HttpURLConnection connection = this.openConnection(parsedUrl1, request);
        Iterator responseCode = map.keySet().iterator();

        while(responseCode.hasNext()) {
            String protocolVersion = (String)responseCode.next();
            connection.addRequestProperty(protocolVersion, (String)map.get(protocolVersion));
        }

        setConnectionParametersForRequest(connection, request);
        ProtocolVersion protocolVersion1 = new ProtocolVersion("HTTP", 1, 1);
        int responseCode1 = connection.getResponseCode();
        if(responseCode1 == -1) {
            throw new IOException("Could not retrieve response code from HttpUrlConnection.");
        } else {
            BasicStatusLine responseStatus = new BasicStatusLine(protocolVersion1, connection.getResponseCode(), connection.getResponseMessage());
            BasicHttpResponse response = new BasicHttpResponse(responseStatus);
            response.setEntity(entityFromConnection(connection));// 这句才是真正的网络请求,返回结果后赋予response
            Iterator var12 = connection.getHeaderFields().entrySet().iterator();

            while(var12.hasNext()) {
                Entry header = (Entry)var12.next();
                if(header.getKey() != null) {
                    BasicHeader h = new BasicHeader((String)header.getKey(), (String)((List)header.getValue()).get(0));
                    response.addHeader(h);
                }
            }

            return response;
        }
    }

2.6 ExecutorDelivery

在通过HurlStack获取HttpResponse结果后,我们返回到BasicNetwork继续读代码,发现将HttpResponse转换成NetworkResponse并且返回到NetworkDispatcher,最后执行了mDelivery.postResponse(request, volleyError1)分发结果。

    public void postResponse(Request<?> request, Response<?> response, Runnable runnable) {
        request.markDelivered();
        request.addMarker("post-response");
        //调用了自定义的Runnable,其实是将子线程切换到了主线程
        this.mResponsePoster.execute(new ExecutorDelivery.ResponseDeliveryRunnable(request, response, runnable));
    }

    private class ResponseDeliveryRunnable implements Runnable {
        private final Request mRequest;
        private final Response mResponse;
        private final Runnable mRunnable;

        public ResponseDeliveryRunnable(Request request, Response response, Runnable runnable) {
            this.mRequest = request;
            this.mResponse = response;
            this.mRunnable = runnable;
        }

        public void run() {
            if(this.mRequest.isCanceled()) {// 请求取消
                this.mRequest.finish("canceled-at-delivery");
            } else {
                if(this.mResponse.isSuccess()) {
                    this.mRequest.deliverResponse(this.mResponse.result);// 分发结果给request,request最终分发给监听器
                } else {
                    this.mRequest.deliverError(this.mResponse.error);
                }

            }
        }
    }

2.7 CacheDispatcher

上面已经完成了一个完整的网络请求并且将结果分发给监听器,供开发者使用结果。在之前我们先讲解了NetworkDispatcher,并且也看到了mCache.put(request.getCacheKey(), volleyError1.cacheEntry),将缓存实体存入缓存对象中。CacheDispatcher也是一个线程,循环来取缓存队列里的请求:

    public void run() {
        //省略部分代码...
        this.mCache.initialize();// 缓存类初始化,主要是在SD卡里创建缓存文件夹
        while(true) {
            final Request e = (Request)this.mCacheQueue.take();// 拿出缓存请求
            e.addMarker("cache-queue-take");
            if(e.isCanceled()) {// 请求取消
                e.finish("cache-discard-canceled");
            } else {
                Cache.Entry entry = this.mCache.get(e.getCacheKey());// 根据key拿到缓存实体
                if(entry == null) {
                    e.addMarker("cache-miss");
                    this.mNetworkQueue.put(e);// 如果没有缓存则加入网络请求队列进行网络请求
                } else if(entry.isExpired()) {// 过期的也要加入网络请求队列
                    e.addMarker("cache-hit-expired");
                    e.setCacheEntry(entry);
                    this.mNetworkQueue.put(e);
                } else {
                    e.addMarker("cache-hit");
                    Response response = e.parseNetworkResponse(new NetworkResponse(entry.data, entry.responseHeaders));// 将NetworkResponse转换成Response
                    e.addMarker("cache-hit-parsed");
                    if(entry.refreshNeeded()) {// 新鲜度判断,如果需要刷新则在分发结果的同时加入网络请求队列,去获取最新的结果
                        e.addMarker("cache-hit-refresh-needed");
                        e.setCacheEntry(entry);
                        response.intermediate = true;
                        this.mDelivery.postResponse(e, response, new Runnable() {
                            public void run() {
                                try {
                                    CacheDispatcher.this.mNetworkQueue.put(e);
                                } catch (InterruptedException var2) {
                                    ;
                                }

                            }
                        });
                    } else {
                        this.mDelivery.postResponse(e, response);// 分发结果(和NetworkDispatcher一样)
                    }
                }
            }
        }
    }

三、总结

随着一点点的跟进代码,我们理清了Volley的工作流程:
1.初始化网络请求类(BasicNetwork)、缓存类(DiskBasedCache)等,创建请求队列(RequestQueue)并启动缓存请求线程(CacheDispatcher)和网络请求线程(NetworkDispatcher )。
2.CacheDispatcher和NetworkDispatcher不断轮询取出请求,如果有缓存则进行过期时间、新鲜度等判断,来决定是从Cache中取缓存还是再去进行网络请求。网络请求通过BasicNetwork调用HttpStack来返回请求结果后会进行缓存(除非你自己设置不进行缓存)。
3.最后通过ExecutorDelivery实现类分发给我们创建的Request中,而根据我们使用的Request(StringRequest、JsonRequest、ImageRequest等)返回对应的数据结构。
以上就是Volley的主要工作原理,我觉得先看懂主要原理是重要的一步,理解原理后再去看代码细节,比如Volley中的PoolingByteArrayOutputStream(继承ByteArrayOutputStream并给二进制输出流增加缓存区)、HttpHeaderParser(用来解析返回结果的头部)等。

参考:

http://a.codekk.com/detail/Android/grumoon/Volley%20%E6%BA%90%E7%A0%81%E8%A7%A3%E6%9E%90

最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容