Retrofit源码 - 流程解析

本文基于Retrofit2.5.0分析

    compile 'com.squareup.retrofit2:retrofit:2.5.0'

Retrofit源码用到的主要思想是 动态代理+注解:
首先我们来看Retrofit的基本用法:
1、首先有个接口,接口里面有注解标记的方法

interface IApiService{
    @GET("client_config.json")
    fun getClientConfigModel(): Call<ClientConfigModel>
}

2、使用create方法获取到上面定义的接口的实例,

val retrofit = Retrofit.Builder()
        .baseUrl(Constants.URL_BASE_OP_PRE)
        .build()
val impl:IApiService = retrofit.create(IApiService::class.java)

3、直接使用,调用方法

RetrofitFactory.impl.getClientConfigModel(url).execute()

上面是Retrofit的基本用法,这里高级API不铺开了。

下面从入口开始看

public static final class Builder {
    private final Platform platform; //当前是什么平台Android或者java
    private @Nullable okhttp3.Call.Factory callFactory; //这里看到其实Retrofit跟OKHttp是深度耦合的,无法拆开。
    private @Nullable HttpUrl baseUrl; //baseUrl
//这里看到converter 和CallAdapter都是可以设置多个的
    private final List<Converter.Factory> converterFactories = new ArrayList<>();
    private final List<CallAdapter.Factory> callAdapterFactories = new ArrayList<>();
    private @Nullable Executor callbackExecutor;
    private boolean validateEagerly;

    Builder(Platform platform) {
      this.platform = platform;
    }

    public Builder() {
      this(Platform.get());
    }

    Builder(Retrofit retrofit) {
      //copy target retrofit的属性到自己
    }

    public Builder client(OkHttpClient client) {
      return callFactory(checkNotNull(client, "client == null"));
    }

    public Builder callFactory(okhttp3.Call.Factory factory) {
      this.callFactory = checkNotNull(factory, "factory == null");
      return this;
    }

    public Builder baseUrl(String baseUrl) {
      checkNotNull(baseUrl, "baseUrl == null");
      return baseUrl(HttpUrl.get(baseUrl));
    }
··· 
    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<CallAdapter.Factory> callAdapterFactories = new ArrayList<>(this.callAdapterFactories);
      callAdapterFactories.addAll(platform.defaultCallAdapterFactories(callbackExecutor));

      // Make a defensive copy of the converters.
      List<Converter.Factory> converterFactories = new ArrayList<>(
          1 + this.converterFactories.size() + platform.defaultConverterFactoriesSize());

      // 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());
      converterFactories.addAll(this.converterFactories);
      converterFactories.addAll(platform.defaultConverterFactories());

      return new Retrofit(callFactory, baseUrl, unmodifiableList(converterFactories),
          unmodifiableList(callAdapterFactories), callbackExecutor, validateEagerly);
    }
  }

Retrofit类的静态内部类Builder,构造器模式构造需要的参数,并构造真正发起网络的OKHttpClient。
然后调用create方法,也就是动态代理方法

public <T> T create(final Class<T> service) {
    ...
    return (T) Proxy.newProxyInstance(service.getClassLoader(), new Class<?>[] { service },
        new InvocationHandler() {
          private final Platform platform = Platform.get();
          private final Object[] emptyArgs = new Object[0];

          @Override public Object invoke(Object proxy, Method method, @Nullable 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);
            }
            return loadServiceMethod(method).invoke(args != null ? args : emptyArgs);
          }
        });
  }

对于动态代理技术,可以参考我的一篇文章,最简单的动态代理技术
有了动态代理的基础后我们发现上面的代码块是一个实现了泛型的动态代理,也就是你给我什么方法我就按照我的逻辑执行什么方法并且返回对应的返回值给你
我们看最重要的方法其实是

return loadServiceMethod(method).invoke(args != null ? args : emptyArgs);

这句话是最终去做事情的一句话,我们跟下去,首先看loadServiceMethod(method)

ServiceMethod<?> loadServiceMethod(Method method) {
    ServiceMethod<?> result = serviceMethodCache.get(method);
    if (result != null) return result;

    synchronized (serviceMethodCache) {
      result = serviceMethodCache.get(method);
      if (result == null) {
        result = ServiceMethod.parseAnnotations(this, method);
        serviceMethodCache.put(method, result);
      }
    }
    return result;
  }

这里首先做了个缓存,用了

  private final Map<Method, ServiceMethod<?>> serviceMethodCache = new ConcurrentHashMap<>();

然后用了synchronized做了个多线程处理,这个HashMap中存的是什么呢?是Method跟ServiceMethod的Key-Value,那ServiceMethod是做什么的呢?

abstract class ServiceMethod<T> {
  static <T> ServiceMethod<T> parseAnnotations(Retrofit retrofit, Method method) {
    RequestFactory requestFactory = RequestFactory.parseAnnotations(retrofit, method);
//获取返回类型
    Type returnType = method.getGenericReturnType();
    if (Utils.hasUnresolvableType(returnType)) {
      throw methodError(method,
          "Method return type must not include a type variable or wildcard: %s", returnType);
    }
    if (returnType == void.class) {
      throw methodError(method, "Service methods cannot return void.");
    }

    return HttpServiceMethod.parseAnnotations(retrofit, method, requestFactory);
  }

  abstract T invoke(Object[] args);
}

我们根据名字猜测这个是去找我们的写的那些@GET @POST的注解了。
RequestFactory其实就是来做这个事情的
RequestFactory的代码就不贴了,太多了,大家可以去看一下RetrofitFactory的代码,就是去规定了那些@GET@POST怎么用的一些规则,包括参数的格式啊,哪些参数应该有body哪些没有body,怎么取内容啊之类的,一堆的if-else。然后存了一些请求的路径端口号什么的。
我们继续看HttpServiceMethod.parseAnnotations
这个时候实际上已经把该准备的东西都准备好了,调用parseAnnotations的时候我们直接看HttpServiceMethod代码

final class HttpServiceMethod<ResponseT, ReturnT> extends ServiceMethod<ReturnT> {
  static <ResponseT, ReturnT> HttpServiceMethod<ResponseT, ReturnT> parseAnnotations(
      Retrofit retrofit, Method method, RequestFactory requestFactory) {
//构造callAdapter
    CallAdapter<ResponseT, ReturnT> callAdapter = createCallAdapter(retrofit, method);
//获取response的类型
    Type responseType = callAdapter.responseType();
    ...
//获取Response转化器
    Converter<ResponseBody, ResponseT> responseConverter =
        createResponseConverter(retrofit, method, responseType);

    okhttp3.Call.Factory callFactory = retrofit.callFactory;
    return new HttpServiceMethod<>(requestFactory, callFactory, callAdapter, responseConverter);
  }

  private static <ResponseT, ReturnT> CallAdapter<ResponseT, ReturnT> createCallAdapter(
      Retrofit retrofit, Method method) {
    Type returnType = method.getGenericReturnType();
    Annotation[] annotations = method.getAnnotations();
    try {
      //noinspection unchecked
      return (CallAdapter<ResponseT, ReturnT>) retrofit.callAdapter(returnType, annotations);
    } catch (RuntimeException e) { // Wide exception range because factories are user code.
      throw methodError(method, e, "Unable to create call adapter for %s", returnType);
    }
  }

  private static <ResponseT> Converter<ResponseBody, ResponseT> createResponseConverter(
      Retrofit retrofit, Method method, Type responseType) {
    Annotation[] annotations = method.getAnnotations();
    try {
      return retrofit.responseBodyConverter(responseType, annotations);
    } catch (RuntimeException e) { // Wide exception range because factories are user code.
      throw methodError(method, e, "Unable to create converter for %s", responseType);
    }
  }

  private final RequestFactory requestFactory;
  private final okhttp3.Call.Factory callFactory;
  private final CallAdapter<ResponseT, ReturnT> callAdapter;
  private final Converter<ResponseBody, ResponseT> responseConverter;

  private HttpServiceMethod(RequestFactory requestFactory, okhttp3.Call.Factory callFactory,
      CallAdapter<ResponseT, ReturnT> callAdapter,
      Converter<ResponseBody, ResponseT> responseConverter) {
    this.requestFactory = requestFactory;
    this.callFactory = callFactory;
    this.callAdapter = callAdapter;
    this.responseConverter = responseConverter;
  }

  @Override ReturnT invoke(Object[] args) {
    return callAdapter.adapt(
        new OkHttpCall<>(requestFactory, args, callFactory, responseConverter));
  }
}

parseAnnotations的时候返回一个HttpServiceMethod对象,里面存储了request跟response需要的工具,然后我们应该还记得最上面动态代理的最后一句

return loadServiceMethod(method).invoke(args != null ? args : emptyArgs);

会调用

@Override ReturnT invoke(Object[] args) {
    return callAdapter.adapt(
        new OkHttpCall<>(requestFactory, args, callFactory, responseConverter));
  }

得到response后对返回结果做处理返回,整个流程就完事儿了,这是大致的流程。后续分析下面继续。


这里是更深入一些的分析。
我们从HttpServiceMethod里面看其实一共没几个方法,除了parseAnnotations和invoke,就是createCallAdapter和createResponseConverter了。顾名思义,这两个方法是构造了callAdapter和converter。
大家常用的CallAdapterConverter大概就是RxJavaCallAdapterFactoryGsonConverterFactory了,
我理解的calladapter应该是在call之前我们要做的准备工作,像RxJavaCallAdapterFactory把RxJava的东西转化成retrofit需要的数据。
而Converter则是把请求回来的数据转化成我们想要的对象或者String类型数据。像GsonConverterFactory就是把我们请求回来的String数据转化成我们需要的对象。

下面我们看源码

retrofit.responseBodyConverter(responseType, annotations);

像createResponseConverter其实就是调用了responseBodyConverter

public <T> Converter<ResponseBody, T> responseBodyConverter(Type type, Annotation[] annotations) {
    return nextResponseBodyConverter(null, type, annotations);
  }
  public <T> Converter<ResponseBody, T> nextResponseBodyConverter(
      @Nullable Converter.Factory skipPast, Type type, Annotation[] annotations) {
...
  int start = converterFactories.indexOf(skipPast) + 1;
    for (int i = start, count = converterFactories.size(); i < count; i++) {
      Converter<ResponseBody, ?> converter =
          converterFactories.get(i).responseBodyConverter(type, annotations, this);
      if (converter != null) {
        return (Converter<ResponseBody, T>) converter;
      }
    }
...
  }
 converterFactories.add(new BuiltInConverters());
      converterFactories.addAll(this.converterFactories);
      converterFactories.addAll(platform.defaultConverterFactories());

最上面build Retrofit的时候有这么一段话,所以这里start就是为了跳过前面添加的。因为Factory.requestBodyConverter 没有实现的时候是返回的null,所以定义的convert的responseBodyConverter如果能匹配上返回类型不返回null,则获取到这个Converter直接返回。所以说如果有两个相同类型的Converter能够匹配同样的ResponseType,那只会取前面的那个,后面的永远到不了。
所谓的ResponseType就是我们请求的时候的响应类型,比如最上面我们请求的
fun getClientConfigModel(): Call<ClientConfigModel>
ResponseType就是ClientConfigModel。所以如果我们使用的时候设置了两个converter

            retrofit = new Retrofit.Builder()
                    .addConverterFactory(new MyConverterFactory())
                    .addConverterFactory(GsonConverterFactory.create())
                    .addCallAdapterFactory(new MyCallAdapterFactory())
                    .baseUrl("https://restapi.amap.com/v3/weather/")
                    .build();

MyConverterFactory和GsonConverterFactory顺序是很关键的,我们可以看GsonConverterFactory的源码,它的responseBodyConverter方法如下:

@Override
  public Converter<ResponseBody, ?> responseBodyConverter(Type type, Annotation[] annotations,
      Retrofit retrofit) {
    TypeAdapter<?> adapter = gson.getAdapter(TypeToken.get(type));
    return new GsonResponseBodyConverter<>(gson, adapter);
  }

我们可以看出来它其实cover了所有的类型。那其实如果是下面这个顺序的话

                    .addConverterFactory(GsonConverterFactory.create())
                    .addConverterFactory(new MyConverterFactory())

我们的MyConverterFactory能够实现什么类型都没有用,因为GsonConverterFactory已经全部截胡了,我们MyConverterFactory没有任何的出场机会。但反过来说下面这个顺序的话

                    .addConverterFactory(new MyConverterFactory())
                    .addConverterFactory(GsonConverterFactory.create())

就是先判断MyConverterFactory的类型,如果MyConverterFactory不能接受我们的返回类型在上面

for (int i = start, count = converterFactories.size(); i < count; i++) {
      Converter<ResponseBody, ?> converter =
          converterFactories.get(i).responseBodyConverter(type, annotations, this);
      if (converter != null) {
        return (Converter<ResponseBody, T>) converter;
      }
    }

就会返回null,然后继续往下for循环,发现GsonConverterFactory可以搞定,就会返回GsonConverterFactory。所以说我们如果需要特殊处理某种类型的话,一定要把它放到前面add进来。否则就是舔到最后一无所有。
我们如果看过OKHttp的源码,印象很深的应该是责任链模式,也就是interceptor拦截器,我感觉retrofit的这个应该也是有异曲同工之妙的~
那我们看CallAdapter的处理是类似的,相对来说Converter的自定义的机会多一些,我们就讲Converter了。
而且我们看invoke方法

@Override ReturnT invoke(Object[] args) {
    return callAdapter.adapt(
        new OkHttpCall<>(requestFactory, args, callFactory, responseConverter));
  }

其实就是调用了callAdapter.adapt方法,而且这个名字起的特别好,adapt就是改编、适应的意思,顾名思义啊,calladapter的作用就是把我们调用的时候的一些东西像RxJava这样的框架的数据改编成我们OKHttp所需要的数据。
我们看OkHttpCall这个类,继承自Call

final class OkHttpCall<T> implements Call<T> {
  private final RequestFactory requestFactory;
  private final Object[] args;
  private final okhttp3.Call.Factory callFactory;
  private final Converter<ResponseBody, T> responseConverter;
  private volatile boolean canceled;
  private @Nullable okhttp3.Call rawCall;
  private @Nullable Throwable creationFailure;
  private boolean executed;

  OkHttpCall(RequestFactory requestFactory, Object[] args,
      okhttp3.Call.Factory callFactory, Converter<ResponseBody, T> responseConverter) {...}

  @Override public synchronized Request request() {...}
  @Override public void enqueue(final Callback<T> callback){...}
  @Override public synchronized boolean isExecuted(){...}
  @Override public Response<T> execute() throws IOException {...}
  Response<T> parseResponse(okhttp3.Response rawResponse) throws IOException {...}
  public void cancel()  {...}
  @Override public boolean isCanceled(){...}
public interface Call<T> extends Cloneable {
  Response<T> execute() throws IOException;
  void enqueue(Callback<T> callback);
  boolean isExecuted();
  void cancel();
  boolean isCanceled();
  Call<T> clone();
  Request request();
}

我们发现Retrofit的Call接口几乎就是把OKHttp的Call接口拷贝过来了。所以Retrofit的OKhttpCall就是把OKHttp的请求包了一遍。就是从okhttp3.Request变成了retrofit2.Request。

下篇写个简单的CallAdapter跟Converter的例子

总结:

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

推荐阅读更多精彩内容

  • 目录介绍 1.首先回顾Retrofit简单使用方法 2.Retrofit的创建流程源码分析2.1 Retrofit...
    杨充211阅读 1,053评论 0 16
  • 一、什么是Retrofit A type-safe HTTP client for Android and Jav...
    andcoder阅读 766评论 2 3
  • 序言 做Android开发的小伙伴都知道,现在最流行的网络框架就是RxJava+Retrofi+OkHttp。今天...
    左大人阅读 1,150评论 1 7
  • 使用 Retrofit 引入 retrofit 使用示例 retrofit 在使用时,需要定义一个接口对象 1. ...
    kevenZheng阅读 247评论 0 1
  • 出来好像三天了吧… 思念我相公,相公一个人在北京,我跑杭州来潇洒啦…其实很想跟你在一起的每个时间段,但生活总会有分...
    呆萌的阿喵小姐阅读 171评论 0 0