Retrofit原理简析

记得很久以前,写过一篇文章,主要说的是Retrofit应用的:Retrofit2+RxJava2踩坑

今天我们来简单说说Retrofit的原理,其实嘛,作为一个搞技术的人,对Retrofit应该有一种好奇的态度,毕竟这是个很优秀的框架,里面用了不少设计模式,很值得我们学习,我们先从Request说起吧。

笔者之前也写过一篇文章:Volley源码学习,就算不看Volley的源码,使用过Volley的人都知道,其实我们真正用的时候会根据请求类型的不同,创建出不同的Request对象,然后将Request对象投放到RequestQueue请求队列中去,然而当你使用Retrofit时,居然连Request对象都不用创建了,这也太神奇了吧!

我们来看看Retrofit到底是如何实现的?
通常我们都会创建一个interface(比如BaseApiService),里面会写各个请求的接口,然后是这么调用的:

BaseApiService apiService = retrofit.create(BaseApiService.class)

然后我们来看看源码:

 public <T> T create(final Class<T> 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();
          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);
          }
        });
  }

如果你对Java的代理模式比较熟悉的话, 当你看到Proxy.newProxyInstance时,你瞬间就会明白:哦,原来用的是Java动态代理!(插个题外话,搞过JavaWeb的应该对这个很熟悉,大名鼎鼎的Spring框架就是用Java动态代理来优雅地实现AOP编程的,因其具有方法增强、高扩展性等优点,故Java动态代理是框架的常用手段)
如此说来,上面返回的BaseApiService对象其实是一个动态代理对象而已。
我们注意到上面invoke方法中的loadServiceMethod,它的返回值是ServiceMethod,其实我们使用了动态代理,最终的目的就是为了获取ServiceMethod对象,我们直接来看看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);
}

这是一个非常简单的抽象类,第一步,很显然目的是为了获取RequestFactory,一看这个名字,我们就知道里面用了工厂模式建造者模式,其实这一步主要就是根据method的各个注解进行拼凑成一个Request,当然其中也按照Retrofit的规则进行了很多的校验。
然后我们看到上面的返回值是一个HttpServiceMethod对象,它又继承于ServiceMethod,所以最终是调用的invoke方法是HttpServiceMethod的invoke方法,我们来看看HttpServiceMethod的构造函数和invoke方法:

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

终于看到OkHttp的名字了,看样子Retrofit的底层是OkHttp所言非虚啊!
的确,Retrofit2中Call接口的默认实现是OkHttpCall,这话的意思就它默认的是OkHttpCall,当然你完全可以根据自己的需要来实现Call接口(也就是不一定要用OkHttp,当然一般情况下我们还是用OkHttp),该点的设计与Volley的HttpStack接口很相似,Volley默认提供了两个实现类:HurlStack和HttpClientStack,当然,我们也可以通过实现HttpStack接口使用OkHttp作为Volley底层的网络框架,这就是面向接口编程的好处

再注意到上面的callAdapter和responseConverter,是不是也很熟悉啊?我们使用Retrofit时一般会与RxJava搭配使用,然后返回的结果我们会用Gson进行转换,相关的代码如下:

 .addConverterFactory(GsonConverterFactory.create())
 .addCallAdapterFactory(RxJava2CallAdapterFactory.create())

好啦,大概也就是这样子吧,事实又一次证明:静下心来分析源码,总能学到东西的,加油!

参考链接:https://blog.csdn.net/csdn_aiyang/article/details/80692384

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

推荐阅读更多精彩内容