Volley 源码分析

1.概述

Volley的源代码分析,网上一大把的分析博客,但是当你看完这些博客后,也是云里雾里的,很多一上来就上源代码分析,典型的就是这一篇:

http://blog.csdn.net/guolin_blog/article/details/17656437

还有从结构上讲,总的来说,比上从源代码讲强那么一点,典型的就是这一篇。单总的来说,这一篇比上一篇强一点。

http://www.codekk.com/open-source-project-analysis/detail/Android/grumoon/Volley%20%E6%BA%90%E7%A0%81%E8%A7%A3%E6%9E%90)

对于一上来就开始分析源代码的行为,我敬而远之。

现在回过头来看Volley的源代码,感觉从设计的角度来说,Volley设计并不算是优秀,但是volley的扩展性非常棒,面向接口编程在这里得到完美的体现,这是阅读volley源代码之后的最大感受。

volley的类没有几个,对于打基础的阶段,阅读该库对提升技术水平有很好的帮助。

volley说明

image
image

从名字由来和配图中无数急促的火箭可以看出 Volley 的特点:特别适合数据量小,通信频繁的网络操作。

对于文件上传,文件下载等操作,还是换成另外的库吧。至于为什么,因为其设计就决定了。

源码分析方法

image
image

其实Volley的设计思路和源码分析的方式在上面的这个图就已经高度的概括展示出来了。这张图已经说明了volley的工作原理和设计思路。

这是是volley源代码的结构:

inage

源代码分析

我们先不看 "toolbox"文件夹,因为该文件夹主要是其接口的实现。

现在我们看下面一些无关紧要的类:

  • VolleyLog。

是一个日志类,与此无关,去掉。

  • VolleyError,TimeOutError,ServerError,ParseError,NetworkError,AuthFailureError,NoConnectionError。

这些都是异常的错误处理,其中后面的Error相关的类都时VolleyError的子类,这个对分析volley结构没有什么帮助,去掉不看。

  • RetryPolicy ,DefaultRetryPolicy。

后者是前者的子类,我们只看RetryPolicy这个就行了。实际上这个是是重置访问,比如本次访问超时,设置20s后再次连接,这个就是重置的作用,对于分析Volley的整个结构来说,无关紧要,可以不看。在分析具体的代码和细节的时候再去看。

  • ResponseDelivery 和 ExecutorDelivery。

后者是前者的一个实现。这个类作用就是异步操作,实际上就是一个Hanlder的事件分发,将子线的结果分发到主线程中。分析结构的时候,只需要关心ResponseDelivery。

  • Cache 缓存的接口。

去掉上面那那些无关紧要的类之后,剩下就没有几个类了,我们看看还剩下那些类。

  • Network 具体的网络访问

  • CacheDispatcher 缓存调度线程

  • NetworkDispatcher 网络访问调度线程

  • NetworkResponse 网络访问的直接结果,不是分发到主线程的结果

  • Request 网络请求的封装,包括请求的url,请求的方法,请求的的内容等等。

  • RequestQueue 这个看起来好像是一个请求的对象,实际上这个只是将这些类根据规则组合在一起的东西。源代码的分析就是从这里从这个类开始进行的。

  • Response 封装了分发的结果,这个怎么理解呢,就是 NetworkRespond的结果经过进一步处理分发到主线程中的结果。

怎么样,剩下的也没有几个,这样就大大的降低了阅读代码的难度了。

RequestQueue开始

这个类是整个Volley的开始部分,先看start()方法:


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();

}

}

前面说过了,CacheDispatcher和NetworkDispatcher是两个线程一个是缓存调度线程,一个是网络请求调度的线程。

上面的代码就是开启一个缓存调度线程和若干个(默认四个)网络请求调度线程。

接下来我们看看CacheDispatcher线程。

CacheDispatcher线程

既然是一个线程,那么我们只关心 线程run方法就可以了

下面是伪代码,剔除一些日志,异常判断等无影响代码后的结构


while(true){

//先从缓存队列中拿出一个请求

final Request request = mCacheQueue.take();

//如果这个请求取消了,那么这个请求直接结束,开始下一个循环

if (request.isCanceled()) {

request.finish("cache-discard-canceled");

ontinue;

}

// 如果没有缓存或者缓存失效的话,就将这条请求添加到网络请求调度线程中

Cache.Entry entry = mCache.get(request.getCacheKey());

if (entry == null || entry.isExpired()) {

mNetworkQueue.put(request);

continue;

}

//从缓存中拿出结果。

Response response = request.parseNetworkResponse(

new NetworkResponse(entry.data, entry.responseHeaders));

request.addMarker("cache-hit-parsed");

//分发结果

mDelivery.postResponse(request, response);

}

从上面的伪代码可以看出,实际上缓存调度线程做的事情非常的简单,逻辑一点也不复杂。

  • 先从缓存队列中拿出一条请求。

  • 判断这个请求是否已经标志结束了,如果结束了那么设置这条请求结束,并分发结果。否则进入下一步。

  • 判断缓存是否存在,缓存是否过期,如果过期了,之前没有缓存,那么就将这条请求放到网络调度的线程中。

NetworkDispatcher

网络请求调度线程的分析方法和上面类似,实际上,该类的run()方法中的代码比CacheDispatcher 线程中 run()方法中代码还少,无非就是:

  • 从网络请求队列中拿出一条request。

  • 判断这条请求是否已经结束或者取消了。如果没有那么就是下一步。

  • 开始执行网络请求,并拿到请求的结果。

  • 是否需要缓存结果,如果需要缓存结果,那么就缓存结果。

  • 将请求的结果进行分发。

完成上面的步骤之后,实际上volley的核心代码就分析完成了,其它部分就是细节的问题,比如具体的网络请求是怎么样子的。分发是什么样子的,但这些都是细节问题,不是主干代码。

对于初学者来说,很容易陷入代码汪洋中,即使把代码全部看一篇也无济于事,抓住枝叶,不见主干,分析再多也没有用。典型的例子有:

http://blog.csdn.net/guolin_blog/article/details/17656437

总结:

阅读源代码的方式我总结如下:

  • 首先看一看设计的结构图。

  • 去掉无关紧要的类,比如异常Error相关的类,日志类等。

  • 分析主干代码,比如重点看接口和抽象类而忽略掉其具体的实现,因为优秀的开源库一定是面向接口编程,这些接口和抽象类才是设计的框架所在。

  • 在分析过程中去掉无用的代码,这里的无用指的是那些为了框架稳定性和环境适配的代码,如NUll指针判断,日志记录,异常等。

  • 使用伪代码自己实现一遍框架。

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

推荐阅读更多精彩内容

  • 前言 本文是一篇日常学习总结性的文章,笔者通过分析经典网络框架Volley的源码,望以巩固Android网络框架中...
    Armstrong_Q阅读 559评论 0 7
  • Volley源码分析之流程和缓存 前言 Android一开始提供了HttpURLConnection和HttpCl...
    大写ls阅读 566评论 0 6
  • 我的博客: Volley 源码分析 Volley 的使用流程分析 官网示例 创建一个请求队列 RequestQue...
    realxz阅读 2,009评论 1 11
  • 第一步:核算时间(Time) (一)天数统计 1001-1007 共7天 本月暂时不安排周末休息,所以可用天数为7...
    成长路上的碎碎念阅读 179评论 0 0
  • 农历十月二十,阴 今天写字心得 1老师的指点是必要的 有时自己的惯性思维会让你局限,老师一点也许就能豁然开朗 2练...
    玲萍阅读 171评论 0 0