OkHttp从原理到应用

为什么我们要采用Okhttp作为网络请求框架

在java 和Android中我们通常采用HttpClient和Httpurlconnection来实现网络请求
现在曾经火爆的网络框架AsyncHttp和Volley (均基于HttpClient和Httpurlconnection)已然淡出我们的视野
Okhttp直接使用Scoket遵循网络协议实现的网络框,这是和其他开源网络框架最大的区别

Okhttp凭什么称霸Android网络框架?

  • 域名解析,数据传输,连接复用,路由选择,代理选择,自动遍历可用服务器节点,缓存,自动重试,等等可控,可高度定制化网络请求。能够低成本实现复杂的业务需求
  • OkHttp 和 Retrofit,Glide,Fresco 第三方库能很好的衔接,拥有良好的生态系统
  • Android4.4的源码中可以看到HttpURLConnection已经替换成OkHttp实现

下面我们一起来探索OkHttp的部分原理和在项目实战中的花式运用

连接复用

Okhttp和服务器建立连接后默认保持5分钟不断开,支持5个socket连接并发,也就是五分钟内客户端如果和已连接的服务器通信不需要重新三次握手连接(三次握手确保了服务端和客户端都具备可靠的通信能力,但握手过程耗时)

HTTP Keep-Alive
在Http早期,每个http请求都要求打开一个tpc socket连接,并且使用一次之后就断开这个tcp连接
使用keep-alive可以改善这种状态,即在一次TCP连接中可以持续发送多份数据而不会断开连接。通过使用keep-alive机制,可以减少tcp连接建立次数,

当使用Keep-Alive模式(又称持久连接、连接重用)时,Keep-Alive功能使客户端到服务器端的连接持续有效,当出现对服务器的后继请求时,Keep-Alive功能避免了建立或者重新建立连接


图片来源于网络

Okhttp中连接复用正是建立在Keep-alive基础之上实现的

Okhttp连接复用实在建立连接过程中使用读写ConnectionPool 中的连接

  public final class ConnectionPool { 
  //保持连接时间
  private final long keepAliveDurationNs;
  //清理超时过期连接的runable
  private final Runnable cleanupRunnable = new Runnable() {
    @Override public void run() {
      while (true) {
        long waitNanos = cleanup(System.nanoTime());
        if (waitNanos == -1) return;
        if (waitNanos > 0) {
          long waitMillis = waitNanos / 1000000L;
          waitNanos -= (waitMillis * 1000000L);
          synchronized (ConnectionPool.this) {
            try {
              ConnectionPool.this.wait(waitMillis, (int) waitNanos);
            } catch (InterruptedException ignored) {
            }
          }
        }
      }
    }
  };

 public ConnectionPool() {
    //默认保持5分钟长连接
    this(5, 5, TimeUnit.MINUTES);
  }

  public ConnectionPool(int maxIdleConnections, long keepAliveDuration, TimeUnit timeUnit) {
    this.maxIdleConnections = maxIdleConnections;
    this.keepAliveDurationNs = timeUnit.toNanos(keepAliveDuration);

    // Put a floor on the keep alive duration, otherwise cleanup will spin loop.
    if (keepAliveDuration <= 0) {
      throw new IllegalArgumentException("keepAliveDuration <= 0: " + keepAliveDuration);
    }
  }
}

Okhttp在打开网络连接时会读写连接池中保存的可用连接,以达到复用
假定客户端和服务器已建立连接,那么在有效期内客户端再次和服务端通信则不需要再次建立连接

连接复用流程图:

连接复用机制

DNS

DNS(Domain Name System,域名系统),DNS 服务用于在网络请求时,将域名转为 IP 地址。能够使用户更方便的访问互联网,而不用去记住能够被机器直接读取的 IP 数串。


image.png

传统的基于 UDP 协议的公共 DNS 服务极易发生 DNS 劫持,从而造成安全问题。

Okhttp中的DNS

Okhttp中使用Dns.lookup将域名解析成ip,默认使用系统解析dns
如果一个域名绑定多个ip,即部署了多个服务节点。Okhttp在建立连接过程中会遍历所有ip,直至建立一个可用连接
Okhttp在ConnectInterceptor调用StreamAllocation打开连接是时会遍历所有ip直接建立一个可靠的链接
实现流程如下:

Okhttp DNS

掌中通智能切换服务器

掌中通先使用的服务端中,有一个域名是带cdn的,有一个是没有的,那么我们如何做到在一个域名挂了自动切换到另一个节点呢?
回到上面的话题,访问业务服务需要同于域名访问,最终是通过服务器ip建立连接进而访问,那么我们回到Okhttp源码
简单的说okhttp建立连接的时候会遍历dns.lookup 查找出来的ip直到建立可用连接位置。
那么我们只需要在dns.lookup的时候,将带cdn和不带cdn的服务url解的域名解析出ip,合并成一个数组既可以以最低成本实现这个需求。

自定义 OKhttp Dns解析源码如下

public class CndAutoSwitchDns implements Dns {


    @Override
    public List<InetAddress> lookup(String hostname) throws UnknownHostException {
        if (hostname.contains(getCdnUrl()) || hostname.contains(BASE_NOCDN_URL_ADD)) {
            boolean userServer2 = ShareManager.getInstance().getBoolean(Constants.USE_SERVER_TWO);
            if (userServer2) {
                return getInetAddressByIps(BASE_NOCDN_URL_ADD, hostname);
            } else {
                return getInetAddressByIps(hostname, BASE_NOCDN_URL_ADD);
            }
        }
        return Dns.SYSTEM.lookup(hostname);
    }

}

HttpDns

客户端解析DNS的过程,由客户端发出解析,容易被运营商等中间层劫持,存在劫持和低效的问题。
HttpDNS 使用HTTP协议进行域名解析,代替现有基于UDP的DNS协议,域名解析请求直接发送到HTTPDNS服务器,从而绕过运营商的Local DNS,能够避免Local DNS造成的域名劫持问题和调度不精准问题


image.png
  public class HttpDns implements Dns {

    @Override
    public List<InetAddress> lookup(String hostname) throws UnknownHostException {
        //从自己的DNS服务更加域名查询IP
        List<String> serverIpList = DnsUtil.getIpByHost(hostname);
        //如果没有查询到则使用系统解析dns
        if (serverIpList == null || serverIpList.isEmpty()) {
            return Dns.SYSTEM.lookup(hostname);
        }
        //合并转换成查询到的ip
        List<InetAddress> inetAddressList = new ArrayList<>();
        for (String ip : serverIpList) {
            inetAddressList.add(InetAddress.getByName(ip));
        }
        return inetAddressList;
    }  
}

使用过程如下非常简单

        OkHttpClient.Builder builder = new OkHttpClient.Builder();
        builder.dns(new HttpDns());
        OkHttpClient mOkHttpClient = builder.build();

如何防止charles,fiddler抓包

反编译,代理抓包,这些开发和测试中常用的技巧。charles在配置https证书后手机设置代理后,手机app中的https和http请求基本上属于裸奔
那么我们怎么设置让我们的app没那么容易被抓包呢?
charles https抓包原理,charles伪装成服务器和手机建立http是连接,中间人攻击劫持,收到客户端发出消息后和目标服务器建立连接,作为中间人和服务器通信所以可以抓包。


image.png

okhttp 默认使用系统路由选择器,默认跟随系统设置,如果手机设置代理app请求也会走代理
原理上很简单在创建Okhttpclient的时候设置一个自定义路由选择器,自定义的路由选择器设置成Proxy.NO_PROXY(不走代理)

 class SafeProxySelector : ProxySelector() {
    override fun select(uri: URI?): MutableList<Proxy> {
        return Collections.singletonList(Proxy.NO_PROXY)
    }

    override fun connectFailed(uri: URI?, sa: SocketAddress?, ioe: IOException?) {
    }
}

Token刷新

目前中通几乎所有的系统基本上都已对接安全系统,授权信息过期后可用刷新替换新的授权信息
登录过程如下:


使用拦截器对所有请求中添加授权相关信息
发起请求后,拦截返回报文状态码,如果是授权信息过期状态,则拦截请求,刷新授权信息,刷新成功后
使用新授权信息重发请求
请求流程流程图:

刷新token拦截器

class RefreshTokenInterceptor : Interceptor {

    companion object {
        private const val TOKEN = "Token"
        private const val OPEN_ID = "OpenId"
        private const val REFRESH_CODE = 406
    }

    override fun intercept(chain: Interceptor.Chain): Response? {
        val request = chain.request()
         if (request.header(TOKEN)!=null){
            return chain.proceed(request)
        }
        //如果本地没有token则不处理
        val localToken = Sso.getToken()

        if (isTokenNullOrEmpty(localToken)) {
            Sso.refreshTokenListener?.refreshFailure()
            return chain.proceed(request)
        }
        //添加本地token到请求header
        val newRequest = updateRequest(request, localToken!!)
        val newResponse = chain.proceed(newRequest)
        //根据状态码判断是否需要刷新token
        if (!needRefresh(newResponse)) {
            return newResponse
        }
        //网络刷新token
        val refreshSsoInfo = refreshToken(localToken)
        if (refreshSsoInfo==null) {
            //重新登录
            Sso.refreshTokenListener?.refreshFailure()
            return newResponse
        }

        //刷新token header重新发送网络请求
        val refreshRequest = updateRequest(newRequest, refreshSsoInfo)
        return chain.proceed(refreshRequest)
    }

    private fun needRefresh(newResponse: Response): Boolean {
        return newResponse.code() == REFRESH_CODE
    }


    /**
     * token为空空校验
     */
    private fun isTokenNullOrEmpty(token: TokenData?): Boolean {
        if (token == null) {
            return true
        }
        if (TextUtils.isEmpty(token.access_token)) {
            return true
        }
        return false
    }

    @Synchronized
    fun refreshToken(@NotNull oldTokenData: TokenData): TokenData? {
        val localToken = Sso.getToken()

        if (localToken!=null && (localToken!!.access_token != oldTokenData.access_token)) {
            return localToken
        }
        try {
            val tokenDataResponse: retrofit2.Response<TokenData> = Sso.refreshToken(oldTokenData.refresh_token).execute()
            if (tokenDataResponse.isSuccessful) {
                val serverSsoInfoEntity = tokenDataResponse.body()
                if (!isTokenNullOrEmpty(serverSsoInfoEntity)) {
                    //保存最新token到本地
                    Sso.saveToken(serverSsoInfoEntity!!)
                    return  serverSsoInfoEntity
                }
             }
        } catch (e: IOException) {
            e.printStackTrace()
        }
        return null
    }

    private fun updateRequest(request: Request, ssoInfoEntity: TokenData): Request {
        return request
                .newBuilder()
                .header(TOKEN, ssoInfoEntity.access_token)
                .header(OPEN_ID, ssoInfoEntity.openid).build()

    }


}

android源码中是如何使用OkHttp作为网络请求框架

2013年Google发布Android 4.4 时,Httpurlconnection 底层采用了OkHttp实现,然而怎么实现的呢

我们来看看这段非常普通创建http连接的代码

    URL url = URL("http://www.zto.com/");
    url.openConnection();

顺藤摸瓜打开URL源码(以下是精简版的部分源码)

import java.net.MalformedURLException;
import java.net.URLStreamHandler;

public class URL {

    //URLStreamHandler 是所有流协议处理程序的通用超类,
    transient URLStreamHandler handler;


    public URL(String protocol, String host, int port, String file,
               URLStreamHandler handler) throws MalformedURLException {
        //构造函数中穿件URLStreamHandler
        if (handler == null && (handler = getURLStreamHandler(protocol)) == null) {
            throw new MalformedURLException("unknown protocol: " + protocol);
        }
        this.handler = handler;
    }

    /**
     * 根据协议获取URLStreamHandler
     * @param protocol
     * @return
     */
    static URLStreamHandler getURLStreamHandler(String protocol) {
        URLStreamHandler handler = null;
        try {
            handler = createBuiltinHandler(protocol);
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }

        return handler;

    }


    /**
     * 创建URLStreamHandler的具体方法
     * @param protocol
     * @return
     * @throws ClassNotFoundException
     * @throws InstantiationException
     * @throws IllegalAccessException
     */
    private static URLStreamHandler createBuiltinHandler(String protocol)
            throws ClassNotFoundException, InstantiationException, IllegalAccessException {
        URLStreamHandler handler = null;
        if (protocol.equals("file")) {
            handler = new sun.net.www.protocol.file.Handler();
        } else if (protocol.equals("ftp")) {
            handler = new sun.net.www.protocol.ftp.Handler();
        } else if (protocol.equals("jar")) {
            handler = new sun.net.www.protocol.jar.Handler();
        } else if (protocol.equals("http")) {
            //如果是http请求或者https请求创建用Okhttp实现的URLStreamHandler
            handler = (URLStreamHandler) Class.
                    forName("com.android.okhttp.HttpHandler").newInstance();
        } else if (protocol.equals("https")) {
            handler = (URLStreamHandler) Class.
                    forName("com.android.okhttp.HttpsHandler").newInstance();
        }
        return handler;
    }

    /**
     * 打开http连接
     * @return
     * @throws java.io.IOException
     */
    public URLConnection openConnection() throws java.io.IOException {
        return handler.openConnection(this);
    }
}

Android源码中使用的并非最新版本的Okhttp 所以,我们看这个分支源码

package com.squareup.okhttp;
import java.io.IOException;
import java.net.Proxy;
import java.net.URL;
import java.net.URLConnection;
import java.net.URLStreamHandler;

public final class HttpHandler extends URLStreamHandler {
    @Override
    protected URLConnection openConnection(URL url) throws IOException {
        return new OkHttpClient().open(url);
    }

    @Override
    protected URLConnection openConnection(URL url, Proxy proxy) throws IOException {
        if (url == null || proxy == null) {
            throw new IllegalArgumentException("url == null || proxy == null");
        }
        return new OkHttpClient().setProxy(proxy).open(url);
    }

    @Override
    protected int getDefaultPort() {
        return 80;
    }
}


public final class OkHttpClient implements URLStreamHandlerFactory{
     public HttpURLConnection open(URL url) {
          return open(url, proxy);
     }

  HttpURLConnection open(URL url, Proxy proxy) {
    String protocol = url.getProtocol();
    OkHttpClient copy = copyWithDefaults();
    copy.proxy = proxy;

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

推荐阅读更多精彩内容