博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Retrofit 源码剖析-深入
阅读量:6163 次
发布时间:2019-06-21

本文共 7418 字,大约阅读时间需要 24 分钟。

背景

前一章节,先系统的讲解了关于Retrofit实现当中的主要技术动态代理,本篇详细结合动态代理在Retrofit中的应用,扩展到结合RxJava来使用

思路

要深入研究一款开源项目最好的入口就是它所暴露出来的外部使用接口,按照这个思路,所以需要大体先了解Retrofit的基本使用,这里就不阐述这些基础的知识,可以查看以前的博客

项目分析

想要弄清楚 Retrofit 的细节,先来看一下 Retrofit 源码的组成:

这里写图片描述
  • 一个 retrofit2.http 包,里面全部是定义 HTTP 请求的注解,比如 GET、POST、PUT、DELETE、Headers、Path、Query 等等

  • 余下的 retrofit 2.0 包中十几个类和接口就是全部 retrofit 的代码了,代码真的很少,很简单,因为retrofit把网络请求这部分功能全部交给了 okHttp 了

初始对象

先上一段使用Retrofit开始前的初始代码

//手动创建一个OkHttpClient并设置超时时间缓存等设置        OkHttpClient.Builder builder = new OkHttpClient.Builder();        builder.connectTimeout(basePar.getConnectionTime(), TimeUnit.SECONDS);        /*创建retrofit对象*/        Retrofit retrofit = new Retrofit.Builder()                .client(builder.build())                .addConverterFactory(ScalarsConverterFactory.create())                .addCallAdapterFactory(RxJavaCallAdapterFactory.create())                .baseUrl(basePar.getBaseUrl())                .build();复制代码

这里的入口Builder()方法进入源码

public static final class Builder {    private Platform platform;    private okhttp3.Call.Factory callFactory;    private HttpUrl baseUrl;    private List
converterFactories = new ArrayList<>(); private List
adapterFactories = new ArrayList<>(); private Executor callbackExecutor; private boolean validateEagerly; Builder(Platform platform) { this.platform = platform; // Add the built-in converter factory first. This prevents overriding its behavior but also // ensures correct behavior when using converters that consume all types. converterFactories.add(new BuiltInConverters()); } public Builder() { this(Platform.get()); }xxxx复制代码

Builder类中主要是记录请求过程中的一些配置信息,比如基础url,当然重要的是addConverterFactory方法记录Converter数据转换器;addCallAdapterFactory方法记录CallAdapter-HTTP请求返回数据的类型.

Converter数据转换器

Converter数据转换器默认存放在retrofit-converters

这里写图片描述

Converter采用接口抽象的方式,灵活了转换器的种类,常用的如上图中的GsonString,同样这里可以可以自由扩展,实现Converter.Factory接口

CallAdapter请求返回数据的类型

工程位置如图

这里写图片描述

这里同Converter数据转换器思想一样同样是采用抽象接口的方式,继承CallAdapter.Factory类实现扩展,正如上图中箭头指示的显示扩展的RxJava2的扩展

信息的转换

Builder记录完数据后通过build()方法,将收集的信息传递给最后的Retrofit对象中,并且初始了okhttp3.Call.Factory请求处理类,从中可以看出Retrofit2.0底层的http请求默认就是通过okhttp3处理的。

/**     * Create the {
@link Retrofit} instance using the configured values. *

* Note: If neither {

@link #client} nor {
@link #callFactory} is called a default {
@link * OkHttpClient} will be created and used. */ public Retrofit build() { if (baseUrl == null) { throw new IllegalStateException("Base URL required."); } okhttp3.Call.Factory callFactory = this.callFactory; if (callFactory == null) { callFactory = new OkHttpClient(); } Executor callbackExecutor = this.callbackExecutor; if (callbackExecutor == null) { callbackExecutor = platform.defaultCallbackExecutor(); } // Make a defensive copy of the adapters and add the default Call adapter. List
adapterFactories = new ArrayList<>(this.adapterFactories); adapterFactories.add(platform.defaultCallAdapterFactory(callbackExecutor)); // Make a defensive copy of the converters. List
converterFactories = new ArrayList<>(this.converterFactories); return new Retrofit(callFactory, baseUrl, converterFactories, adapterFactories, callbackExecutor, validateEagerly); }复制代码

动态代理

Retrofit初始完成以后,需要调用create方法传入一个java接口抽象类对象作为参数。其实这一步真是Retrofit动态代理生成的地方

源码:

@SuppressWarnings("unchecked") // Single-interface proxy creation guarded by parameter safety.  public 
T create(final Class
service) { Utils.validateServiceInterface(service); if (validateEagerly) { eagerlyValidateMethods(service); } return (T) Proxy.newProxyInstance(service.getClassLoader(), new Class
[] { service }, new InvocationHandler() { private final Platform platform = Platform.get(); @Override public Object invoke(Object proxy, Method method, Object... args) throws Throwable { // If the method is a method from Object then defer to normal invocation. if (method.getDeclaringClass() == Object.class) { return method.invoke(this, args); } if (platform.isDefaultMethod(method)) { return platform.invokeDefaultMethod(method, service, proxy, args); } ServiceMethod serviceMethod = loadServiceMethod(method); OkHttpCall okHttpCall = new OkHttpCall<>(serviceMethod, args); return serviceMethod.callAdapter.adapt(okHttpCall); } }); }复制代码

使用代码:

HttpTestService httpService = retrofit.create(HttpTestService.class);        httpService.getAllVedioBy(isAll());复制代码

上面HttpTestService对象其实是一个动态代理对象,并不是一个真正的 HttpTestService接口的implements对象,当 httpService对象调用 getAllVedioBy方法时,执行的是下面动态代理方法。

ServiceMethod serviceMethod = loadServiceMethod(method);            OkHttpCall okHttpCall = new OkHttpCall<>(serviceMethod, args);            return serviceMethod.callAdapter.adapt(okHttpCall);复制代码

其中主要ServiceMethod是主要的处理对象

ServiceMethod信息类

当调用loadServiceMethod方法,将抽象HttpTestService中通过注解定义的信息收集起来存放于ServiceMethod中,其中主要包含了一下数据:

  • OkHttpClient:发送网络请求的工具

  • RequestFactory: 类似于 Volley 中的 Request,包含了HTTP请求的Url、Header信息,MediaType、Method以及RequestAction数组

  • CallAdapter:HTTP请求返回数据的类型

  • Converter:数据转换器

最后Map<Method, ServiceMethod> serviceMethodCache保存全部的 ServiceMethod信息类

ServiceMethod
loadServiceMethod(Method method) { ServiceMethod
result = serviceMethodCache.get(method); if (result != null) return result; synchronized (serviceMethodCache) { result = serviceMethodCache.get(method); if (result == null) { result = new ServiceMethod.Builder<>(this, method).build(); serviceMethodCache.put(method, result); } } return result; }复制代码

debug可查看当前ServiceMethod对象收集的信息

这里写图片描述

收集完信息以后执行了

OkHttpCall okHttpCall = new OkHttpCall<>(serviceMethod, args);复制代码

这一步对okhttp熟悉的同学肯定知道了,就是通过okhttp请求数据,其中serviceMethod是请求的地址信息,args是当前接口需要的参数信息。

执行完http请求以后,执行了

serviceMethod.callAdapter.adapt(okHttpCall);复制代码

这一步是将okhttp获取到的数据传递给一开始Retrofit通过builder指定的RxJavaCallAdapterFactory类,那它是如何一步步处理的呢?

其实在一开始的获取serviceMethod方法loadServiceMethod(Method method)中,已经将Retrofit对象传递给了serviceMethod对象

核心代码:

//这里的this即是`Retrofit`对象result = new ServiceMethod.Builder(this, method).build();复制代码

通过.Builder(this, method).build()方法将Retrofit一开始的builder信息全部传递给了现在的serviceMethod对象,所以当执行

serviceMethod.callAdapter.adapt(okHttpCall);复制代码

serviceMethod.callAdapter便制动转换成了RxJavaCallAdapterFactory,调用CallAdapter抽象接口adapt进入到RxJavaCallAdapterFactory类中

源码:

@Override public 
Observable
> adapt(Call
call) { Observable
> observable = Observable.create(new CallOnSubscribe<>(call)); if (scheduler != null) { return observable.subscribeOn(scheduler); } return observable; } }复制代码

最后生成RxJavaObservable对象,后面通过得到的Observable对象就是对数据的处理了。

到此基于Retrofit结合RxJava源码的解读就完成了,回头看整个过程其实还是很简单,作者条理清晰理解起来也很简单,一口气终于把Retrofit大体撸了一遍。

总结

Retrofit 非常巧妙的用注解来描述一个 HTTP 请求,将一个 HTTP 请求抽象成一个 Java 接口,然后用了 Java 动态代理的方式,动态的将这个接口的注解“翻译”成一个 HTTP 请求,最后再执行这个 HTTP 请求

Retrofit的功能非常多的依赖 Java 反射,代码中其实还有很多细节,比如异常的捕获、抛出和处理,大量的 Factory 设计模式。

Retrofit 中接口设计的恰到好处,在你创建 Retrofit 对象时,让你有更多更灵活的方式去处理你的需求,比如使用不同的 Converter、使用不同的 CallAdapter,这也就提供了你使用 RxJava 来调用 Retrofit 的可能。

专栏

源码

建议

转载地址:http://reefa.baihongyu.com/

你可能感兴趣的文章
谈缓存和Redis
查看>>
【转】百度地图api,根据多点注标坐标范围计算地图缩放级别zoom自适应地图
查看>>
用户调研(补)
查看>>
ExtJS之开篇:我来了
查看>>
☆1018
查看>>
oracle 去掉空格
查看>>
6.13心得
查看>>
Runtime类
查看>>
eclipse decompiler
查看>>
记一个搜索网盘资源的网站
查看>>
jdk1.7和jdk1.8的String的getByte方法的差异
查看>>
java父子进程通信
查看>>
Android ADB server didn't ACK * failed to start daemon * 简单有效的解决方案
查看>>
Olap学习笔记
查看>>
Codeforces Round #431 (Div. 1)
查看>>
如何进行数组去重
查看>>
将标题空格替换为 '_' , 并自动复制到剪切板上
查看>>
List Collections sort
查看>>
Mysql -- You can't specify target table 'address' for update in FROM clause
查看>>
使用局部标准差实现图像的局部对比度增强算法。
查看>>