Android 网络: Volley+OkHttp+Https

本文介绍如何使用OkHttp替代Volley的默认Http传输层,通过OkHttp的特性提升网络请求效率,包括SPDY支持、连接池、GZIP压缩、响应缓存和IP多址处理。同时,针对HTTPS安全问题,提供了将自签名证书打包入APK的方法,确保应用在使用HTTPS时的安全性和稳定性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

Volley 默认根据 Android 系统版本使用不同的 Http 传输协议实现. 
在 Android 3.0 以上 Volley 使用 ApacheHttpStack 作为传输协议, 在2.3 及以下使用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();

  1. 这样就行了.

作为一个有节操的开发者应该使用 Https 来保护用户的数据, Android 开发者网站上文章Security with HTTPS and SSL做了详尽的阐述.

OkHttp 自身是支持 Https 的. 参考文档 OkHttp Https, 直接使用上面的 OkHttpStack就可以了, 但是如果遇到服务器开发哥哥使用了自签名的证书(不要问我为什么要用自签名的), 就无法正常访问了.

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

我采用的方案是将自签名的证书打包入 情侣QQ网名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);
     }
 }
}
<pre name="code" class="html" style="color: rgb(85, 85, 85); font-size: 13px; line-height: 24px;"><span style="font-family: 'Lantinghei SC', Arial, 'Microsoft YaHei', sans-serif;font-size:14px;">然后用 </span><code style="word-wrap: break-word; box-sizing: border-box; outline: none; font-size: 1em; margin: 0px 0.2em; word-break: break-word; background-color: rgb(248, 249, 250);">SelfSignSslOkHttpStack</code><span style="font-family: 'Lantinghei SC', Arial, 'Microsoft YaHei', sans-serif;font-size:14px;"> 创建 Volley 的 RequestQueue.</span>
 
 
<pre name="code" class="html">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();


 


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值