( 二 ) Volley 源码深入了解之 Dispatcher

一、概述

从上一篇博客中我们已经了解了 RequestQueue ,如果对 RequestQueue 不是很了解那么不妨先看看上篇博客 ( 一 ) Volley 源码深入了解之 RequestQueue,本篇博客将会接着上篇的分析继续从源码的角度深入了解 CacheDispatcher 和 NetworkDispatcher 是如何工作的

二、 深入了解 CacheDispatcher

在分析之前,照列先看官方给的流程图


image.png

经过上篇的分析创建 request 添加到 RequestQueue 过程我们已经很清楚了,而两个 Dispatcher 内部的工作流程我们还没有很深入的了解,这一节我们先看看 CacheDispatcher

上一篇我们说过,当调用 RequestQueue 的 start 方法时会创建 CacheDispatcher 和 NetworkDispatcher 对象 ,现在我们来回顾一下

public class RequestQueue {
   /** 启动所有 dispatcher */
    public void start() {
        stop(); // 启动之前先停止所有 dispathcer
        // 在这里创建了缓存调度器并启动
        mCacheDispatcher = new CacheDispatcher(mCacheQueue, mNetworkQueue, mCache, mDelivery);
        mCacheDispatcher.start();

        // 在这里启动网络调度器线程池(数量默认为4)并全部启动
        for (int i = 0; i < mDispatchers.length; i++) {
            NetworkDispatcher networkDispatcher =
                    new NetworkDispatcher(mNetworkQueue, mNetwork, mCache, mDelivery);
            mDispatchers[i] = networkDispatcher;
            networkDispatcher.start();
        }
    }

    /** 停止调度 */
    public void stop() {
        if (mCacheDispatcher != null) {
            mCacheDispatcher.quit();//调用 Thread 的 interrupt 方法终止线程
        }
        //终止线程池
        for (final NetworkDispatcher mDispatcher : mDispatchers) {
            if (mDispatcher != null) {
                mDispatcher.quit();//同上
            }
        }
    }
    //省略若干代码.....
}

可以看到先创建了 CacheDispatcher 对象并启动了,我们直接进入 CacheDispatcher 中看看

public class CacheDispatcher extends Thread {
    
    //省略无关代码
    
    /** 缓存队列 */
    private final BlockingQueue<Request<?>> mCacheQueue;

    /** 网络请求队列 */
    private final BlockingQueue<Request<?>> mNetworkQueue;

    /** 要读取的缓存 */
    private final Cache mCache;

    /** 用于发送获取 response 到主线程 */
    private final ResponseDelivery mDelivery;

    /** 线程终止标记. */
    private volatile boolean mQuit = false;

    /**管理等待中的请求和避免相同的缓存 key 进行重复的请求 */
    private final WaitingRequestManager mWaitingRequestManager;

    /**
     *  CacherDispatcher 的唯一构造方法
     */
    public CacheDispatcher(
            BlockingQueue<Request<?>> cacheQueue,
            BlockingQueue<Request<?>> networkQueue,
            Cache cache,
            ResponseDelivery delivery) {
        //上面的注释已经说明,不在赘述
        mCacheQueue = cacheQueue; 
        mNetworkQueue = networkQueue; 
        mCache = cache; 
        mDelivery = delivery;
        mWaitingRequestManager = new WaitingRequestManager(this);
    }
    //省略若干代码.......
}

可以看到 CacherDispatcher 继承自 Thread ,那就好办了我们直接找 run 方法就行了

public class CacheDispatcher extends Thread {
    
    //省略无关代码
  
   @Override
    public void run() {
        if (DEBUG) VolleyLog.v("start new dispatcher");
        //设置线程的优先级,默认为后台线程
        Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND);

        // 初始化缓存(线程安全的方法)
        mCache.initialize();
        //死循环,保证一直在后台轮询,取出缓存队列中的 request 处理
        while (true) {
            try {
                //处理 request
                processRequest();
            } catch (InterruptedException e) {
                // We may have been interrupted because it was time to quit.
                if (mQuit) {
                    Thread.currentThread().interrupt();
                    return;
                }
            }
        }
    }

    //省略若干代码.......
}

run 方法中首先对缓存进行了初始化,然后调用了处理 request 的方法我们跟进看看

   private void processRequest() throws InterruptedException {
        // 从缓存队列中取出 request
        final Request<?> request = mCacheQueue.take();
        processRequest(request);
    }

   /**
     * 这个方法对缓存的各种状态进行判断并处理
     */
    void processRequest(final Request<?> request) throws InterruptedException {
        request.addMarker("cache-queue-take");
        request.sendEvent(RequestQueue.RequestEvent.REQUEST_CACHE_LOOKUP_STARTED);

        try {
            // 如果这个请求已被 cancel 则结束请求并停止该方法的执行
            if (request.isCanceled()) {
                request.finish("cache-discard-canceled");
                return;
            }

            // 通过 request 的 cache key 恢复当前 reqeust 的 cache 信息
            Cache.Entry entry = mCache.get(request.getCacheKey());
            if (entry == null) {//缓存为 null
                request.addMarker("cache-miss");
                // Cache miss; send off to the network dispatcher.
                //判断当前的 request 是否被添加到了等待列表中
                if (!mWaitingRequestManager.maybeAddToWaitingRequests(request)) {
                    //没有缓存,那就加入的网络请求队列中
                    mNetworkQueue.put(request);
                }
                return;
            }

            // 恢复的 request 缓存是否过期
            if (entry.isExpired()) {
                request.addMarker("cache-hit-expired");
                request.setCacheEntry(entry);
                if (!mWaitingRequestManager.maybeAddToWaitingRequests(request)) {
                    mNetworkQueue.put(request);//过期且不再等待列表中则加入到网络请求队列
                }
                return;
            }

            // 如果上面的判断都没有问题,则解析缓存信息并传递这个请求到 main thread
            request.addMarker("cache-hit");
            Response<?> response =
                    request.parseNetworkResponse(
                            new NetworkResponse(entry.data, entry.responseHeaders));
            request.addMarker("cache-hit-parsed");
            //这里通过 softTll 来判断是否过期
            if (!entry.refreshNeeded()) {
                //没有过期则直接返回 response
                mDelivery.postResponse(request, response);
            } else {
                // 这里的处理就比较有意思了,如果过期了依然会先返回缓存的response
                // 然后在把这个 request 加入网络请求队列,这就是上一节中 softTtl 的作用
                request.addMarker("cache-hit-refresh-needed");
                request.setCacheEntry(entry);
                // Mark the response as intermediate.
                response.intermediate = true;
                //可以看到这里面的逻辑正是前面说的
                if (!mWaitingRequestManager.maybeAddToWaitingRequests(request)) {
                    // Post the intermediate response back to the user and have
                    // the delivery then forward the request along to the network.
                    mDelivery.postResponse(
                            request,
                            response,//缓存 response
                            new Runnable() {
                                @Override
                                public void run() {
                                    try {
                                        mNetworkQueue.put(request);//加入网络请求队列
                                    } catch (InterruptedException e) {
                                        // Restore the interrupted status
                                        Thread.currentThread().interrupt();
                                    }
                                }
                            });
                } else {
                    // request has been added to list of waiting requests
                    // to receive the network response from the first request once it returns.
                    mDelivery.postResponse(request, response);
                }
            }
        } finally {//发送请求完成的事件
            request.sendEvent(RequestQueue.RequestEvent.REQUEST_CACHE_LOOKUP_FINISHED);
        }
    }

代码很长,但并不复杂,可以从中看到 processRequest 方法对出队的 request 进行了非常严谨的处理和判断,包括取消状态,缓存是否为null, 是否过期,是否在等待的请求列表中等等,但是总体来说干的事情并不复杂,说的直白一点就是出队,判断请求是否可用,可用则直接传递到主线程,不可用则加入到网络请求队列,只是中间做了完善的处理。不过有两个地方需要深入探究一下,看看下面这行代码

mWaitingRequestManager.maybeAddToWaitingRequests(request)

这行代码多次出现,前面也有对它的简单解释,但它到底干了什么呢? 它是如何处理重复请求的呢? 想搞清楚的话,那么就得深入研究一下 WaitingRequestManager 这个类了,先让我们看下这个类的基本结构

private static class WaitingRequestManager implements Request.NetworkRequestCompleteListener {
        //这个map的作用相当于是对于重复请求的一个暂存区域
        private final Map<String, List<Request<?>>> mWaitingRequests = new HashMap<>();

        private final CacheDispatcher mCacheDispatcher;
        //与 CacheDispathcer 关联
        WaitingRequestManager(CacheDispatcher cacheDispatcher) {
            mCacheDispatcher = cacheDispatcher;
        }
        
        @Override //处理接收到的 response
        public void onResponseReceived(Request<?> request, Response<?> response) {}

        @Override //处理不可用的无效的 response
        public synchronized void onNoUsableResponseReceived(Request<?> request) {
        //处理等待中的请求,一般就是重复的请求
        private synchronized boolean maybeAddToWaitingRequests(Request<?> request) {}
}

注释中已对各个字段和方法做了一个简短说明,但这里还是要说明一下,WaitingRequestManager 这个类是 CacheDispathcer 的私有内部类,实现了 Request 的内部接口 NetworkRequestCompleteListener 并重写了两个方法,如上面的代码所示,可以看到只有maybeAddToWaitingRequests 是该内部类的自有方法,我们看看它做了什么

      private synchronized boolean maybeAddToWaitingRequests(Request<?> request) {
            String cacheKey = request.getCacheKey();//获取传入的 request 的 cache key 
            //判断这个请求是否已经在集合中
            if (mWaitingRequests.containsKey(cacheKey)) {
                // 如果有则加入暂存列表其实就是重复的请求列表
                List<Request<?>> stagedRequests = mWaitingRequests.get(cacheKey);
                if (stagedRequests == null) {
                    stagedRequests = new ArrayList<>();
                }
                request.addMarker("waiting-for-response");
                stagedRequests.add(request);
                mWaitingRequests.put(cacheKey, stagedRequests);
                if (VolleyLog.DEBUG) {
                    VolleyLog.d("Request for cacheKey=%s is in flight, putting on hold.", cacheKey);
                }
                return true; //包含重复的 request 返回 true
            } else {
                // Insert 'null' queue for this cacheKey, indicating there is now a request in
                // flight.
                mWaitingRequests.put(cacheKey, null);
                //如果不包含则与传入的 request 对象关联方便 request 调用
                request.setNetworkRequestCompleteListener(this);
                if (VolleyLog.DEBUG) {
                    VolleyLog.d("new request, sending to network %s", cacheKey);
                }
                return false; //不包含返回 false
            }
        }

可以看到 maybeAddToWaittingRequests 方法内部的逻辑并不复杂,当等待集合中有这个缓存时则 add 到暂存列表中,并返回 true ,否则将等待列表置为 null 且与传入的 Request 进行关联,并返回 false, 这有什么用呢? 其实结合 processRequest 中的代码看,理解这个方法作用还是很简单的,当缓存的 request 已经过期或为null 但又重复请求时,则会被添加到等待的暂存列表中等待被调用,并且把这个 reqeust 加入到网络请求队列中,且结束本次的 CacheDispatcher 轮询,相当于一次 cache miss 也就是没有拿到缓存,没有拿到缓存的话那么就由 NetworkDispatcher 来处理这个 request 了。至于 onResponseReceived 和 onNoUsableResponseReceived 方法这里先卖个关子放到后面讲

分析完 maybeAddToWaittingRequests 后我们来思考一下,缓存中的数据是如何被传递到主线程中从而被我们拿到的呢? 换句话说 Volley 是如何把获得的 response 传递给调用者的呢?

让我们回忆一下我们是从哪里获取的数据

    String url = "https://www.baidu.com";
    RequestQueue mRequestQueue = Volley.newRequestQueue(getApplicationContext());//创建一个请求队列
    StringRequest request = new StringRequest(Request.Method.GET, url, new Response.Listener<String>() {
        @Override
        public void onResponse(String s) {
            Toast.makeText(getApplicationContext(),s,Toast.LENGTH_SHORT).show();
        }, new Response.ErrorListener() {
            @Override
            public void onErrorResponse(VolleyError volleyError) {
                Toast.makeText(getApplicationContext(),volleyError.toString(),Toast.LENGTH_SHORT).show();
            }
        });
    request.setTag("volley");
    mRequestQueue.add(request);
    mRequestQueue.start();

可以看到是在 Respons 的内部接口 Listener 中的 onResponse 中拿到的数据,在前面的分析中似乎没有发现这个接口的调用痕迹,那么 response data 是怎么过来的呢?答案就是通过 ExecutorDelivery 这个对象, 其实 ExecutorDelivery 这个对象在上一篇出现过,如果你看的仔细那么肯定有印象,就是在 4.2.3 深入了解 RequestQueue 小节中的开头部分,在 RequestQueue 的构造函数中被实例化并被传入到 CacheDispathcer 和 NetworkDispacher 中。我们来看看这个类的具体内容

// respinse 传递的接口
public interface ResponseDelivery {
    /** 将从网络或缓存获取的 response 解析并传递 */
    void postResponse(Request<?> request, Response<?> response);

    /**
     * 功能同上,不同的是提供了一个 Runnable 接口在传递 response 后执行任务
     */
    void postResponse(Request<?> request, Response<?> response, Runnable runnable);

    /** 将错误信息 post 给指定的请求. */
    void postError(Request<?> request, VolleyError error);
}

// ResponseDelivery 的具体实现类,用于传递 response 到主线程
public class ExecutorDelivery implements ResponseDelivery {
    //用于在主线程传递 response 的执行接口,该接口下的 excute 方法接收一个 Runnable 接口
    private final Executor mResponsePoster;
    
    /**
     * 接收 Handler 的构造函数
     * @param handler {@link Handler} to post responses on
     */
    public ExecutorDelivery(final Handler handler) {
        // Make an Executor that just wraps the handler.
        mResponsePoster =  new Executor() {//提供了 Executor 的默认实现
                    @Override
                    public void execute(Runnable command) {
                        handler.post(command);
                    }
                };
    }

    /**
     * 接收一个 Executor ,可以用来自定义你的传递方式
     * @param executor For running delivery tasks
     */
    public ExecutorDelivery(Executor executor) {
        mResponsePoster = executor;
    }

    @Override
    public void postResponse(Request<?> request, Response<?> response) {
        postResponse(request, response, null);
    }

    @Override
    public void postResponse(Request<?> request, Response<?> response, Runnable runnable) {
        request.markDelivered();
        request.addMarker("post-response");
        mResponsePoster.execute(new ResponseDeliveryRunnable(request, response, runnable));
    }

    @Override
    public void postError(Request<?> request, VolleyError error) {
        request.addMarker("post-error");
        Response<?> response = Response.error(error);
        mResponsePoster.execute(new ResponseDeliveryRunnable(request, response, null));
    }

    private static class ResponseDeliveryRunnable implements Runnable {
        private final Request mRequest;//传入的 request
        private final Response mResponse;//要传递的 response
        private final Runnable mRunnable;//传递后执行的 Runnable

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

        @SuppressWarnings("unchecked")
        @Override
        public void run() {
            // 如果这个请求被取消则停止请求,并终止传递
            if (mRequest.isCanceled()) {
                mRequest.finish("canceled-at-delivery");
                return;
            }

            // 传递错误或者成功的 response 取决于 isSuccess
            if (mResponse.isSuccess()) {
                mRequest.deliverResponse(mResponse.result);
            } else {
                mRequest.deliverError(mResponse.error);
            }

            // 这是一个中间相应,当 softTll 过期时会调用
            if (mResponse.intermediate) {
                mRequest.addMarker("intermediate-response");
            } else {
                mRequest.finish("done");
            }

            // 这里表示传递完 response 后可以执行一个 Runnable
            if (mRunnable != null) {
                mRunnable.run();
            }
        }
    }
}

可以看到代码并不复杂,ExecutorDelivery 是 ResponseDelivery 接口的具体实现类实现了三个 post 方法,主要用来向主线程传递成功或失败的 response ,而且 ExecutorDelivery 共有两个构造函数,第一个接收一个 Handler 对象来传递 response 默认提供了一个 Excutor 接口的实现。第二个则接收一个 Excutor 接口,用来自定义你的传递方式,其中 ExecutorDelivery 还有一个私有的实现了 Runnable 接口的静态内部类 ResponseDeliveryRunnable 专门用来处理具体传递任务,内部的 run 方法逻辑很简单就不多说了,我们重点看下 mRequest.deliverResponse 方法(这是 Request 对象的方法),从中我们可以看到这行才是真正传递 response 的代码,我们进入 Request 对象看看

public abstract class Request<T> implements Comparable<Request<T>> {
    protected abstract void deliverResponse(T response);

    public void deliverError(VolleyError error) {
        Response.ErrorListener listener;
        synchronized (mLock) {
            listener = mErrorListener;
        }
        if (listener != null) {
            listener.onErrorResponse(error);
        }
    }

    //省略若干代码
}

可以看到 Request 是一个实现了 Comparable 接口的抽象类,这个抽象类主要是对一个 request 的一些共有特性进行了抽离,例如设置 request 的 GET/POST 、Header、request 的参数等等,这些很简单就不多说了,我们就重点看下上面的两个方法,deliverResponse/Error 方法是用来将 response 回调给主线程的,其中 Request 已经提供了默认的 deliveryError 方法,而 deliveryRespose 则需要子类去实现,比如 StringRequest ,JsonRequest 等等。

这里提一下 Response 这个对象, 顾名思义这个对象是对从网络或缓存中获取的 response 的一个封装类,内部定义了两个接口如下列代码

public class Response<T> {

    /** Callback interface for delivering parsed responses. */
    public interface Listener<T> {
        /** Called when a response is received. */
        void onResponse(T response);
    }

    /** Callback interface for delivering error responses. */
    public interface ErrorListener {
        /**
         * Callback method that an error has been occurred with the provided error code and optional
         * user-readable message
         */
        void onErrorResponse(VolleyError error);
    }

    //省略若干代码
}

是不是很眼熟呢,没错这就是我们最终拿到 response / error 信息的两个接口,当调用 deliveryError 的时候其实最终调用的就是 Response 的内部接口 ErrorListener 的 onErrorResponse 方法。 deliveryResponse 也是同理,用的都是一个逻辑,这个可以看下 StringRequest 中的实现,一样的方式,唯一和 error 不同的地方在于调用的是 Response 的内部接口 Listener 的 onResponse 方法(读者可以自己看下 Volley 提供的几个默认 Request 查看 ),这里就不多赘述了。

现在再看下面的代码是不是觉得对内部的工作机制有了一个更为清晰的认知呢

    String url = "https://www.baidu.com";
    RequestQueue mRequestQueue = Volley.newRequestQueue(getApplicationContext());//创建一个请求队列
    StringRequest request = new StringRequest(Request.Method.GET, url, new Response.Listener<String>() {
        @Override
        public void onResponse(String s) {
            Toast.makeText(getApplicationContext(),s,Toast.LENGTH_SHORT).show();
        }, new Response.ErrorListener() {
            @Override
            public void onErrorResponse(VolleyError volleyError) {
                Toast.makeText(getApplicationContext(),volleyError.toString(),Toast.LENGTH_SHORT).show();
            }
        });
    request.setTag("volley");
    mRequestQueue.add(request);
    mRequestQueue.start();

三、深入了解 NetworkDispatcher

NetworkDispatcher 是一个网络请求调度器,当 CacheDispatcher 没有发现缓存信息时,NetworkDispatcher 就会从请求队列中获取 Request 并发起这个网络请求

 /** Starts the dispatchers in this queue. */
    public void start() {
        stop(); // Make sure any currently running dispatchers are stopped.
        // Create the cache dispatcher and start it.
        mCacheDispatcher = new CacheDispatcher(mCacheQueue, mNetworkQueue, mCache, mDelivery);
        mCacheDispatcher.start();

        // Create network dispatchers (and corresponding threads) up to the pool size.
        for (int i = 0; i < mDispatchers.length; i++) {
            NetworkDispatcher networkDispatcher =
                    new NetworkDispatcher(mNetworkQueue, mNetwork, mCache, mDelivery);
            mDispatchers[i] = networkDispatcher;
            networkDispatcher.start();
        }
    }

可以看到在 RequestQueue 中的 start 方法中创建了 NetworkDispatcher 线程池(默认是长度为 4),构造函数中传入的参数和 CacheDispatcher 差不多唯一的区别就是少了一个 CacheQueue ,由于和 CacheDispatcher 的结构差不多,这里就不详细介绍 NetworkDispatcher 的类结构了,我们直接进入 run 方法

    @Override
    public void run() {
        Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND);
        while (true) {//死循环,让线程一直轮询,不断的从请求队列中取出请求
            try {
                processRequest();
            } catch (InterruptedException e) {
                // We may have been interrupted because it was time to quit.
                if (mQuit) {
                    Thread.currentThread().interrupt();
                    return;
                }
                VolleyLog.e(
                        "Ignoring spurious interrupt of NetworkDispatcher thread; "
                                + "use quit() to terminate it");
            }
        }
    }

和 CacheDispathcer 的 run 方法差不多,内部也是声明了一个 processRequest 方法来处理请求,跟进看看

private void processRequest() throws InterruptedException {
        // 从请求队列中获取请求
        Request<?> request = mQueue.take();
        processRequest(request);
}

void processRequest(Request<?> request) {
        long startTimeMs = SystemClock.elapsedRealtime();
        request.sendEvent(RequestQueue.RequestEvent.REQUEST_NETWORK_DISPATCH_STARTED);
        try {
            request.addMarker("network-queue-take");

            // 如果当前 request 已经取消则停止这个 request
            if (request.isCanceled()) {
                request.finish("network-discard-cancelled");
                request.notifyListenerResponseNotUsable();
                return;
            }

            addTrafficStatsTag(request);

            // 执行这个网络请求
            NetworkResponse networkResponse = mNetwork.performRequest(request);
            request.addMarker("network-http-complete");

            //如果服务端返回的数据和上次相同且已经传递过了,则停止重复的请求
            if (networkResponse.notModified && request.hasHadResponseDelivered()) {
                request.finish("not-modified");
                request.notifyListenerResponseNotUsable();
                return;
            }

            // 解析从网络中获取的 response
            Response<?> response = request.parseNetworkResponse(networkResponse);
            request.addMarker("network-parse-complete");

            // 如果这个 request 需要写入缓存,则更新缓存
            if (request.shouldCache() && response.cacheEntry != null) {
                mCache.put(request.getCacheKey(), response.cacheEntry);
                request.addMarker("network-cache-written");
            }

            // 标记这个请求已经传递过
            request.markDelivered();
            //直接将获取的 response 传递给主线程
            mDelivery.postResponse(request, response);
            //一个提示接收 response 的监听
            request.notifyListenerResponseReceived(response);
        } catch (VolleyError volleyError) {
            volleyError.setNetworkTimeMs(SystemClock.elapsedRealtime() - startTimeMs);
            parseAndDeliverNetworkError(request, volleyError);
            request.notifyListenerResponseNotUsable();
        } catch (Exception e) {
            VolleyLog.e(e, "Unhandled exception %s", e.toString());
            VolleyError volleyError = new VolleyError(e);
            volleyError.setNetworkTimeMs(SystemClock.elapsedRealtime() - startTimeMs);
            mDelivery.postError(request, volleyError);
            request.notifyListenerResponseNotUsable();
        } finally {//发送网络请求调用结束的事件
            request.sendEvent(RequestQueue.RequestEvent.REQUEST_NETWORK_DISPATCH_FINISHED);
        }
    }

从上面的代码中可以看到 NetworkDispatcher 和 CacheDispatcher 对于 request 处理其实大同小异,总体的逻辑处理其实是相同的,很好理解,只是侧重点不同。这里面唯一让我感兴趣的是两个 notify 方法,这两个 notify 方法最终会调用 Request.NetworkRequestCompleteListener 的实现类 WaitingRequestManager 中的两个对应的方法, 如下所示

private static class WaitingRequestManager implements Request.NetworkRequestCompleteListener {
      
       @Override //处理接收到的 response
       public void onResponseReceived(Request<?> request, Response<?> response) {}

       @Override //处理不可用的无效的 response
       public synchronized void onNoUsableResponseReceived(Request<?> request) {
}

前面我们已经分析过这个类,并卖了个关子,(如果不清楚可以回顾一下前面关于 maybeAddToWaitingRequests 的分析)下面我们就来逐一看看这两个方法到底干了什么

  • onNoUsableResponseReceived
         // 处理无效的 response 方法
        @Override
        public synchronized void onNoUsableResponseReceived(Request<?> request) {
            String cacheKey = request.getCacheKey();//获取该请求的缓存 key 
            //从缓存的等待集合中移除并获取这个等待的请求列表
            List<Request<?>> waitingRequests = mWaitingRequests.remove(cacheKey);
            if (waitingRequests != null && !waitingRequests.isEmpty()) {
                if (VolleyLog.DEBUG) {
                    VolleyLog.v(
                            "%d waiting requests for cacheKey=%s; resend to network",
                            waitingRequests.size(), cacheKey);
                }
                //从等待的请求列表中移除该 request 并获取下一个
                Request<?> nextInLine = waitingRequests.remove(0);
                //更新map集合
                mWaitingRequests.put(cacheKey, waitingRequests);
                //与下一个 request 进行关联
                nextInLine.setNetworkRequestCompleteListener(this);
                try {//将下一个请求加入到网络请求队列
                    mCacheDispatcher.mNetworkQueue.put(nextInLine);
                } catch (InterruptedException iex) {
                    VolleyLog.e("Couldn't add request to queue. %s", iex.toString());
                    // Restore the interrupted status of the calling thread (i.e. NetworkDispatcher)
                    Thread.currentThread().interrupt();
                    // Quit the current CacheDispatcher thread.
                    mCacheDispatcher.quit();
                }
            }
        }

当 request 调用这个方法时,说明这个 request 已经被取消或是一个重复的但 response 没有发生改变的状态。而这个方法就是用来处理这种情况,当这种情况的 Request 传入该方法后,会
先从 mWaitingRequests 这个 map 集合中取出该缓存 key 的等待 list 并移除,然后就是从中取出下一个请求更新 map 集合,并将其加入到网络请求队列中,其总体的思想就是,把这个无用的请求从等待列表中移除,取出另一个等待中的 request 加入 networkQueue 等待被调用。

  • onResponseReceived
        //处理成功拿到 response 的 request
        @Override
        public void onResponseReceived(Request<?> request, Response<?> response) {
            //没有被缓存过或已经过期,则当做无效的 request 处理
            if (response.cacheEntry == null || response.cacheEntry.isExpired()) {
                onNoUsableResponseReceived(request);
                return;
            }
          //获取缓存 key
            String cacheKey = request.getCacheKey();
            List<Request<?>> waitingRequests;
            synchronized (this) {//从等待的 map 或者也可以理解成重复的请求集合中移除
                waitingRequests = mWaitingRequests.remove(cacheKey);
            }
            if (waitingRequests != null) {
                if (VolleyLog.DEBUG) {
                    VolleyLog.v(
                            "Releasing %d waiting requests for cacheKey=%s.",
                            waitingRequests.size(), cacheKey);
                }
                // 处理所有等待中的请求
                for (Request<?> waiting : waitingRequests) {//
                    mCacheDispatcher.mDelivery.postResponse(waiting, response);
                }
            }
        }

这个方法会在 request 成功拿到 response 后被调用,其实这个方法和 onNoUsableResponseReceived 的方法很相似,也是先把这个 request 从等待的 map 集合中删除并去除对应的 等待列表,唯一不同的是上一个方法时把取出的 request 加入到网络请求队列中,这里是直接全部传递到主线程中去了。刚开始看到这里的时候我是有些疑惑的,因为有好几个地方调用了 mDelivery.postResponse 方法那我们拿的到底是哪一个 response 呢?

其实大可放心,还记得这个方法吗?

 // 标记这个请求已经传递过
 request.markDelivered();

当调用这个方法时,该 request 的 response 就被标记为已传递过了,那么此时 hasHadResponseDelivered 方法返回的就是 true ,这时请求会被 Volley 停止,同时该请求会被 onNoUsableResponseReceived 方法加入到等待列表中和网络请求对列中等待下次轮询被调用。

至此,Volley 官方给出的流程图到这里算是分析完了。

四、结语

经过两篇的源码深入分析,到这就算是结束了。此时再回过头看下官方给的流程图发现理解的更透彻了,当然 Volley 还有很多的细节值得探讨,但我想看的是整体的设计思想,所以更深的细节没有过多深究,也没必要过于深究将自己陷入牛角尖影响学习的心态。而阅读源码就是要从宏观的层面理解作者的设计思想,从而结合到实际运用到工作中,只有达到这样的程度才是真的学以致用!

IT 人员谁不想写出让人称赞的代码呢? 而通往这条路的捷径就是阅读更多更优秀的源码,也只有这样才能更快的掌握编程的 "里子" !就像 Linux 之父 林纳斯·托瓦兹 说的那样,Talk is cheap. Show me the code.

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

推荐阅读更多精彩内容