Volley源码解析使用方式和使用场景分析

概述

Volley是Google在2013年推出的一个网络库,用于解决复杂网络环境下网络请求问题。刚推出的时候是非常火的,现在该项目的变动已经很少了。项目库地址为
https://android.googlesource.com/platform/frameworks/volley
通过提交历史可以看到,最后一次修改距离今天已经有一段时间了。而volley包的release版本也已经很久没有更新了。
Jeff Davidson <jpd@google.com>Sun Mar 13 16:35:59 2016 +0000
虽然很久没有更新了,Volley始终是一个很好的网络框架,我们来分析一下volley的源码,更好的了解volley的使用场景,设计模式,还有存在的一些小问题,或者说使用不当出现的问题。

创建RequestQueue

Volley类实质上只提供了一个方法newRequestQueue,用来创建RequestQueue,RequestQueue是volley的请求队列,mCurrentRequests中存储了执行中的和将要执行的请求,DEFAULT_NETWORK_THREAD_POOL_SIZE是一个常量4。
可以通过RequestQueue 的public RequestQueue(Cache cache, Network network, int threadPoolSize)这个方法修改线程数量,默认开启4个线程,然后一直子后台运行。这里需要注意一下在调用Volley的RequestQueue的时候,内部已经调用了RequestQueue的start方法,不需要再次调用。如果自己创建RequestQueue需要自行调用start方法,整个APP的生命周期中使用一次即可。多次调用会增加线程开销,每次调用start方法,都会调用stop方法终止原来的线程,然后重新开启新的线程。
正常使用volley后台请求线程数量是固定的,默认4个并发不需要修改,可能是基于这个考虑,并没有使用Executor线程池,线程池的考虑本身是为了管理线程频繁创建,避免过多开销的。默认始终4个线程,不存在过度开销问题。个人感觉这里使用线程池会更好一些,当然引入线程池复杂度一定会增加。始终只有4个线程也引发了一些问题,使volley在某些场景不适用。如果请求服务器响应时间太长,4个线程都会处于阻塞状态,这个时候新来的请求只能等待,不能直接执行。volley是比较适合轻量级请求,请求频繁,请求时间短。

    /** Number of network request dispatcher threads to start. */
    private static final int DEFAULT_NETWORK_THREAD_POOL_SIZE = 4;

    public RequestQueue(Cache cache, Network network) {
        this(cache, network, DEFAULT_NETWORK_THREAD_POOL_SIZE);
    }
  Network network = new BasicNetwork(stack);

        RequestQueue queue = new RequestQueue(new DiskBasedCache(cacheDir), network);
        queue.start();

请求执行者HttpStack

HttpStack是真正执行网络请求的接口,performRequest方法执行请求,源码中有两个实现,一个是HurlStack,另一个是HttpClientStack,SDK版本大于等于9使用的是HurlStack。

       if (stack == null) {
            if (Build.VERSION.SDK_INT >= 9) {
                stack = new HurlStack();
            } else {
                // Prior to Gingerbread, HttpUrlConnection was unreliable.
                // See: http://android-developers.blogspot.com/2011/09/androids-http-clients.html
                stack = new HttpClientStack(AndroidHttpClient.newInstance(userAgent));
            }
        }

DefaultHttpClient和它的兄弟AndroidHttpClient都是HttpClient具体的实现类,它们都拥有众多的API,而且实现比较稳定,bug数量也很少。但同时也由于HttpClient的API数量过多,使得我们很难在不破坏兼容性的情况下对它进行升级和扩展,所以目前Android团队在提升和优化HttpClient方面的工作态度并不积极。

HttpURLConnection是一种多用途、轻量极的HTTP客户端,使用它来进行HTTP操作可以适用于大多数的应用程序。虽然HttpURLConnection的API提供的比较简单,但是同时这也使得我们可以更加容易地去使用和扩展它。不过在Android 2.2版本之前,HttpURLConnection一直存在着一些令人厌烦的bug。比如说对一个可读的InputStream调用close方法时,就有可能会导致连接池失效了。那么我们通常的解决办法就是直接禁用掉连接池的功能。Android 2.3版本之前HttpURLConnection存在bug不建议使用,而在Android 2.3版本及以后,HttpURLConnection则是最佳的选择。它的API简单,体积较小,因而非常适用于Android项目。压缩和缓存机制可以有效地减少网络访问的流量,在提升速度和省电方面也起到了较大的作用。

目前来说,我们有一个更好的请求选择okhttp,volley源码中并没有封装它的请求,我们可以自己实现HttpStack接口,在performRequest使用okhttp请求。OkHttp 相较于其它的实现有以下的优点:支持SPDY,允许连接同一主机的所有请求分享一个socket。 如果SPDY不可用,会使用连接池减少请求延迟。 使用GZIP压缩下载内容,且压缩操作对用户是透明的。 利用响应缓存来避免重复的网络请求。 当网络出现问题的时候,OKHttp会依然有效,它将从常见的连接问题当中恢复。 如果你的服务端有多个IP地址,当第一个地址连接失败时,OKHttp会尝试连接其他的地址,这对IPV4和IPV6以及寄宿在多个数据中心的服务而言,是非常有必要的。使用 OkHttp 作为替代是一个很好的选择。

缓存与线程处理

刚才说有4个默认线程是不准确的,是有4个NetworkDispatcher执行网络请求,还有一个CacheDispatcher缓存线程,本地缓存策略需要实现Cache接口,源码中有两个实现DiskBasedCache,NoCache,默认使用的是DiskBasedCache。我们可以根据自己的需要实现Cache接口。DiskBasedCache默认路径是app缓存目录下的volley,默认缓存5M,超出之后会覆盖旧数据。


Request类

Request类的子类相当于volley的输入,是创建请求的时候用的。JsonObjectRequest、JsonArrayRequest用来处理返回是json的数据,StringRequest处理stirng,ImageRequest用来处理图片。


Volley 其实是一个生产者和消费者系统,调用方是生产者,而Volley是消费者。调用方通过RequestQueue 生产Request,而Vollery 消费Request 从而得到Response。那么负责调配这些生产者和消费者的就是Dispatcher,分别是Cache 和 Network 的Dispatcher。





欢迎扫描二维码,一起学习。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值