Volley 已经发布很长时间了, 也已被广泛应用, 相关教程到处都是. 本文只说两个值得注意的地方.

本文讲解部分比较少,
请参阅提供的相关链接. 完整的实现代码在 Github
dodocat/AndroidNetworkDemo
 可能看起来比这里更清晰.

使用 OkHttp 作为传输层的实现.

Volley 默认根据 Android 系统版本使用不同的 Http 传输协议实现.

在 Android 2.3以下使用 ApacheHttpStack 作为传输协议, 在 3.0 及以下使用 HttpURLConnection 作为传输层协议 (感谢评论中指正的朋友).

OkHttp 相较于其它的实现有以下的优点.

  • 支持SPDY,允许连接同一主机的所有请求分享一个socket。
  • 如果SPDY不可用,会使用连接池减少请求延迟。
  • 使用GZIP压缩下载内容,且压缩操作对用户是透明的。
  • 利用响应缓存来避免重复的网络请求。
  • 当网络出现问题的时候,OKHttp会依然有效,它将从常见的连接问题当中恢复。
  • 如果你的服务端有多个IP地址,当第一个地址连接失败时,OKHttp会尝试连接其他的地址,这对IPV4和IPV6以及寄宿在多个数据中心的服务而言,是非常有必要的。

因此使用 OkHttp 作为替代是好的选择.

  1. 首先用 OkHttp 实现一个新的 HurlStack 用于构建 Volley 的 requestQueue.

    public class OkHttpStack extends HurlStack {
    
     private OkHttpClient okHttpClient;
    
     /**
      * Create a OkHttpStack with default OkHttpClient.
      */
     public OkHttpStack() {
         this(new OkHttpClient());
     }
    
     /**
      * Create a OkHttpStack with a custom OkHttpClient
      * @param okHttpClient Custom OkHttpClient, NonNull
      */
     public OkHttpStack(OkHttpClient okHttpClient) {
         this.okHttpClient = okHttpClient;
     }
    
     @Override
     protected HttpURLConnection createConnection(URL url) throws IOException {
         OkUrlFactory okUrlFactory = new OkUrlFactory(okHttpClient);
         return okUrlFactory.open(url);
     }
    }
  2. 然后使用 OkHttpStack 创建新的 Volley requestQueue.

    requestQueue = Volley.newRequestQueue(getContext(), new OkHttpStack());
    requestQueue.start();

    这样就行了.

使用 Https

作为一个有节操的开发者应该使用 Https 来保护用户的数据, Android 开发者网站上文章Security with HTTPS and SSL做了详尽的阐述.OkHttp
自身是支持 Https 的. 参考文档 OkHttp Https, 直接使用上面的 OkHttpStack 就可以了, 但是如果遇到服务器开发哥哥使用了自签名的证书(不要问我为什么要用自签名的), 就无法正常访问了.

网上有很多文章给出的方案是提供一个什么事情都不做的TrustManager 跳过
SSL 的验证, 这样做很容受到攻击, Https 也就形同虚设了.

我采用的方案是将自签名的证书打包入 APK 加入信任.

好处:

  • 应用难以逆向, 应用不再依赖系统的 trust store, 使得 Charles 抓包等工具失效. 要分析应用 API 必须反编译 APK.
  • 不用额外购买证书, 省钱....

缺点:

  • 证书部署灵活性降低, 一旦变更证书必须升级程序.

实现步骤

以最著名的自签名网站12306为例说明

  1. 导出证书

     echo | openssl s_client -connect kyfw.12306.cn:443 2>&1 |  sed -ne '/-BEGIN CERTIFICATE-/,/-END CERTIFICATE-/p' > kyfw.12306.cn.pem
  2. 将证书转为 bks 格式

    下载最新的bcprov-jdk, 执行下面的命令. storepass 是导出密钥文件的密码.

    keytool -importcert -v \
     -trustcacerts \
     -alias 0 \
     -file <(openssl x509 -in kyfw.12306.cn.pem) \
     -keystore $CERTSTORE -storetype BKS \
     -providerclass org.bouncycastle.jce.provider.BouncyCastleProvider \
     -providerpath ./bcprov-jdk16-1.46.jar \
     -storepass asdfqaz
  3. 将导出的 kyfw.bks 文件放入 res/raw 文件夹下.

  4. 创建 SelfSignSslOkHttpStack

    /**
    * A HttpStack implement witch can verify specified self-signed certification.
    */
    public class SelfSignSslOkHttpStack extends HurlStack {
    
     private OkHttpClient okHttpClient;
    
     private Map<String, SSLSocketFactory> socketFactoryMap;
    
     /**
      * Create a OkHttpStack with default OkHttpClient.
      */
     public SelfSignSslOkHttpStack(Map<String, SSLSocketFactory> factoryMap) {
         this(new OkHttpClient(), factoryMap);
     }
    
     /**
      * Create a OkHttpStack with a custom OkHttpClient
      * @param okHttpClient Custom OkHttpClient, NonNull
      */
     public SelfSignSslOkHttpStack(OkHttpClient okHttpClient, Map<String, SSLSocketFactory> factoryMap) {
         this.okHttpClient = okHttpClient;
         this.socketFactoryMap = factoryMap;
     }
    
     @Override
     protected HttpURLConnection createConnection(URL url) throws IOException {
         if ("https".equals(url.getProtocol()) && socketFactoryMap.containsKey(url.getHost())) {
             HttpsURLConnection connection = (HttpsURLConnection) new OkUrlFactory(okHttpClient).open(url);
             connection.setSSLSocketFactory(socketFactoryMap.get(url.getHost()));
             return connection;
         } else {
             return  new OkUrlFactory(okHttpClient).open(url);
         }
     }
    }
  5. 然后用 SelfSignSslOkHttpStack 创建 Volley 的 RequestQueue.

     String[] hosts = {"kyfw.12306.cn"};
     int[] certRes = {R.raw.kyfw};
     String[] certPass = {"asdfqaz"};
     socketFactoryMap = new Hashtable<>(hosts.length);
    
     for (int i = 0; i < certRes.length; i++) {
         int res = certRes[i];
         String password = certPass[i];
         SSLSocketFactory sslSocketFactory = createSSLSocketFactory(context, res, password);
         socketFactoryMap.put(hosts[i], sslSocketFactory);
     }
    
     HurlStack stack = new SelfSignSslOkHttpStack(socketFactoryMap);
    
     requestQueue = Volley.newRequestQueue(context, stack);
     requestQueue.start();
  6. 我们来试一试, 用上一步穿件的 RequestQueue 替换掉原来的, 然后发请求试试.

         StringRequest request = new StringRequest(
                 Request.Method.GET,
                 "https://kyfw.12306.cn/otn/",
                 new Response.Listener<String>() {
                     @Override
                     public void onResponse(String response) {
                         responseContentTextView.setText(response);
                     }
                 },
                 new Response.ErrorListener() {
                     @Override
                     public void onErrorResponse(VolleyError error) {
                         responseContentTextView.setText(error.toString());
                     }
                 });
         RequestManager.getInstance(this).addRequest(request, this);

Android 网络之 Volley+OkHttp+Https的更多相关文章

  1. Android网络框架Volley(体验篇)

    Volley是Google I/O 2013推出的网络通信库,在volley推出之前我们一般会选择比较成熟的第三方网络通信库,如: android-async-http retrofit okhttp ...

  2. Android网络框架Volley

    Volley是Google I/O 2013推出的网络通信库,在volley推出之前我们一般会选择比较成熟的第三方网络通信库,如: android-async-http retrofit okhttp ...

  3. Android网络框架Volley(实战篇)

      之前讲了ym—— Android网络框架Volley(体验篇),大家应该了解了volley的使用,接下来我们要看看如何把volley使用到实战项目里面,我们先考虑下一些问题: 从上一篇来看 mQu ...

  4. Android网络框架-Volley实践 使用Volley打造自己定义ListView

    这篇文章翻译自Ravi Tamada博客中的Android Custom ListView with Image and Text using Volley 终于效果 这个ListView呈现了一些影 ...

  5. ym—— Android网络框架Volley(终极篇)

    转载请注明本文出自Cym的博客(http://blog.csdn.net/cym492224103).谢谢支持! 没看使用过Volley的同学能够,先看看Android网络框架Volley(体验篇)和 ...

  6. ym—— Android网络框架Volley(体验篇)

    VolleyGoogle I/O 2013推出的网络通信库,在volley推出之前我们一般会选择比较成熟的第三方网络通信库,如: android-async-http retrofit okhttp ...

  7. Android 网络框架Volley的使用

    Volley简介 在平时的开发过程中,我们的应用几乎总是在和网络打交道, 在android下的网络编程一般都是基于Http协议的 ,常见的是HttpURLConnection和HttpClient 两 ...

  8. Android 网络框架 volley源码剖析

    转载请注明出处:  http://blog.csdn.net/guolin_blog/article/details/17656437 经过前三篇文章的学习,Volley的用法我们已经掌握的差不多了, ...

  9. Android 网络框架---Volley

    /** * Volley 可以同时请求多个,允许高并发 * 特性: * 1.JSON.图片等的异步下载 * 2.网络请求的排序(Scheduling) * 3.网络请求的优先级处理 * 4.缓存 * ...

随机推荐

  1. 解决C#编译中"csc不是内部或外部命令"的问题

    安装完 VisualStudio 编译环境后,是不能用命令行直接编译写好的csc文件的,如果不配置环境变量,在命令提示符(cmd)中编译扩展名为cs的文件,会出现错误提示"csc不是内部或外 ...

  2. Jmockit之mock特性详解

    本文是Jmockit学习过程中,根据官网所列的工具特性进行解读. 1.调用次数约束(Invocation count constraints) 可以通过调用计数约束来指定预期和/或允许匹配给定期望的调 ...

  3. Springboot security cas源码陶冶-CasAuthenticationFilter

    Springboot security cas整合方案中不可或缺的校验Filter类或者称为认证Filter类,其内部包含校验器.权限获取等,特开辟新地啃啃 继承结构 - AbstractAuthen ...

  4. [Manacher]【学习笔记】

    终于填坑啦......马拉车 课件上说的好短,但是明白了,讲解稍微修改一下抄上行了,比扩展KMP好写多了 求以每个字符为中心的最长回文串的半径.如果要求可以以字符间隙为回文中心,就要在每两个字符之间及 ...

  5. 51Nod 欢乐手速场1 A Pinball[DP 线段树]

    Pinball xfause (命题人)   基准时间限制:1 秒 空间限制:262144 KB 分值: 20 Pinball的游戏界面由m+2行.n列组成.第一行在顶端.一个球会从第一行的某一列出发 ...

  6. web攻击

    一.XSS(跨站脚本攻击) 最常见和基本的攻击WEB网站的方法.攻击者在网页上发布包含攻击性代码的数据.当浏览者看到此网页时,特定的脚本就会以浏览者用户的身份和权限来执行.通过XSS可以比较容易地修改 ...

  7. 网络服务器操作命令telnet

    有些命令是内部的,系统自带的,在装好系统后,就可以随时使用有些命令是系统中没有的,要自己安装一下才能使用,比如你说的telnet,需要安装一下才能使用的.CentOS中用 yum install te ...

  8. 关于Mac设置alias别名访问服务器

    1.首先要安装zsh[链接]robbyrussell/oh-my-zsh 什么是 oh-my-zsh (官网) 两种下载方式 如下图所示,下载安装成功 vi ~/.zshrc发开 打开zsh配置文件 ...

  9. 在CentOS 7中安装Jetty服务器

    Jetty 是一款纯Java的HTTP (Web) 服务器和Java Servlet容器. 通常在更大的网络框架中,Jetty经常用于设备间的通信,而其他Web服务器通常给"人类" ...

  10. PHPUnit-附录 B. 标注

    [http://www.phpunit.cn/manual/5.7/zh_cn/appendixes.annotations.html] 所谓标注,是指某些编程语言中允许加在源代码中的一种特殊格式的语 ...