紧接上文Android进阶:七、Retrofit2.0原理解析之最简流程【上】

一.请求参数整理

我们定义的接口已经被实现,但是我们还是不知道我们注解的请求方式,参数类型等是如何发起网络请求的呢?

这时我们可能应该关注一下ServiceMethod<Object, Object>对象的构建了:

  1. ServiceMethod<Object, Object> serviceMethod =
  2. (ServiceMethod<Object, Object>) loadServiceMethod(method);

主要的逻辑都在这个loadServiceMethod(method)里面,我们看看方法体:

  1. ServiceMethod<?, ?> loadServiceMethod(Method method) {
  2. ServiceMethod<?, ?> result = serviceMethodCache.get(method);
  3. if (result != null) return result;
  4. synchronized (serviceMethodCache) {
  5. result = serviceMethodCache.get(method);
  6. if (result == null) {
  7. result = new ServiceMethod.Builder<>(this, method).build();
  8. serviceMethodCache.put(method, result);
  9. }
  10. }
  11. return result;
  12. }

逻辑很简单,就是先从一个 serviceMethodCache中取ServiceMethod<?, ?>对象,如果没有,则构建ServiceMethod<?, ?>对象,然后放进去serviceMethodCache中,这个serviceMethodCache是一个HashMap:

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

所以构建ServiceMethod<?, ?>对象的主要逻辑还不在这个方法里,应该在new ServiceMethod.Builder<>(this, method).build();里面。这也是个链式调用,一般都是参数赋值,我们先看看Builder<>(this, method)方法:

  1. Builder(Retrofit retrofit, Method method) {
  2. this.retrofit = retrofit;
  3. this.method = method;
  4. this.methodAnnotations = method.getAnnotations();
  5. this.parameterTypes = method.getGenericParameterTypes();
  6. this.parameterAnnotationsArray = method.getParameterAnnotations();
  7. }

果然,这里获取了几个重要的参数:

  • retrofit实例
  • method,接口方法
  • 接口方法的注解methodAnnotations,在retrofit里一般为请求方式
  • 参数类型parameterTypes
  • 参数注解数组parameterAnnotationsArray,一个参数可能有多个注解

    我们再看看build()的方法:
  1. public ServiceMethod build() {
  2. callAdapter = createCallAdapter();
  3. responseType = callAdapter.responseType();
  4. responseConverter = createResponseConverter();
  5. for (Annotation annotation : methodAnnotations) {
  6. parseMethodAnnotation(annotation);
  7. }
  8. if (httpMethod == null) {
  9. throw methodError("HTTP method annotation is required (e.g., @GET, @POST, etc.).");
  10. }
  11. int parameterCount = parameterAnnotationsArray.length;
  12. parameterHandlers = new ParameterHandler<?>[parameterCount];
  13. for (int p = 0; p < parameterCount; p++) {
  14. Type parameterType = parameterTypes[p];
  15. if (Utils.hasUnresolvableType(parameterType)) {
  16. throw parameterError(p, "Parameter type must not include a type variable or wildcard: %s",
  17. parameterType);
  18. }
  19. Annotation[] parameterAnnotations = parameterAnnotationsArray[p];
  20. if (parameterAnnotations == null) {
  21. throw parameterError(p, "No Retrofit annotation found.");
  22. }
  23. parameterHandlers[p] = parseParameter(p, parameterType, parameterAnnotations);
  24. }
  25. return new ServiceMethod<>(this);
  26. }

这个方法挺长的,删了些无关紧要的代码还是很长。首先一开始先获取几个重要对象:callAdapter、responseType和responseConverter,这三个对象都跟最后的结果有关,我们先不管。

看到一个for循环,遍历方法的注解,然后解析:

  1. for (Annotation annotation : methodAnnotations) {
  2. parseMethodAnnotation(annotation);
  3. }
  1. private void parseMethodAnnotation(Annotation annotation) {
  2. if (annotation instanceof DELETE) {
  3. parseHttpMethodAndPath("DELETE", ((DELETE) annotation).value(), false);
  4. } else if (annotation instanceof GET) {
  5. parseHttpMethodAndPath("GET", ((GET) annotation).value(), false);
  6. }
  7. ....

这个方法的方法体我删掉了后面的一部分,因为逻辑都是一样,根据不同的方法注解作不同的解析,得到网络请求的方式httpMethod。但是主要的方法体还是if里面的方法:

  1. private void parseHttpMethodAndPath(String httpMethod, String value, boolean hasBody) {
  2. ....
  3. // Get the relative URL path and existing query string, if present.
  4. int question = value.indexOf('?');
  5. if (question != -1 && question < value.length() - 1) {
  6. // Ensure the query string does not have any named parameters.
  7. String queryParams = value.substring(question + 1);
  8. Matcher queryParamMatcher = PARAM_URL_REGEX.matcher(queryParams);
  9. if (queryParamMatcher.find()) {
  10. throw methodError("URL query string \"%s\" must not have replace block. "
  11. + "For dynamic query parameters use @Query.", queryParams);
  12. }
  13. }
  14. this.relativeUrl = value;
  15. this.relativeUrlParamNames = parsePathParameters(value);
  16. }

逻辑不复杂,就是校验这个value的值 是否合法,规则就是不能有“?”如果有则需要使用@Query注解。最后this.relativeUrl = value;。这个relativeUrl就相当于省略域名的URL,一般走到这里我们能得到的是:users/{name}/repos这样的。里面的“{name}”是一会我们需要赋值的变量

我们继续看刚才的build()方法:

解析完方法的注解之后,需要解析参数的注解数组,这里实例化了一个一维数组:

  1. parameterHandlers = new ParameterHandler<?>[parameterCount];

然后遍历取出参数的类型:

  1. Type parameterType = parameterTypes[p];

取出参数注解:

  1. Annotation[] parameterAnnotations = parameterAnnotationsArray[p];

然后把参数类型、参数注解都放在一起进行解析,解析的结果放到刚才实例化的数组parameterHandlers里面:

  1. parameterHandlers[p] = parseParameter(p, parameterType, parameterAnnotations);

那我们再看看这个方法里做了什么:

  1. private ParameterHandler<?> parseParameter(int p, Type parameterType, Annotation[] annotations) {
  2. ParameterHandler<?> result = null;
  3. for (Annotation annotation : annotations) {
  4. ParameterHandler<?> annotationAction = parseParameterAnnotation(
  5. p, parameterType, annotations, annotation);
  6. }
  7. }

这个方法的主要代码也很简单,解析参数注解,得到一个ParameterHandler<?> annotationAction对象。

那我继续看方法里面的代码。当我们点进parseParameterAnnotation( p, parameterType, annotations, annotation);的源码里面去之后发现这个方法的代码接近500行!但是大部分逻辑类似,都是通过if else判断参数的注解,我们取一段我们刚才的例子相关的代码出来:

  1. if (annotation instanceof Path) {
  2. if (gotQuery) {
  3. throw parameterError(p, "A @Path parameter must not come after a @Query.");
  4. }
  5. if (gotUrl) {
  6. throw parameterError(p, "@Path parameters may not be used with @Url.");
  7. }
  8. if (relativeUrl == null) {
  9. throw parameterError(p, "@Path can only be used with relative url on @%s", httpMethod);
  10. }
  11. gotPath = true;
  12. Path path = (Path) annotation;
  13. String name = path.value();
  14. validatePathName(p, name);
  15. Converter<?, String> converter = retrofit.stringConverter(type, annotations);
  16. return new ParameterHandler.Path<>(name, converter, path.encoded());
  17. }

前面做了一些校验,后面取出注解的名字:name,然后用正则表达校验这个name是否合法。然后构建一个Converter<?, String>对象:

  1. Converter<?, String> converter = retrofit.stringConverter(type, annotations);

点击去看看:

  1. public <T> Converter<T, String> stringConverter(Type type, Annotation[] annotations) {
  2. ....
  3. for (int i = 0, count = converterFactories.size(); i < count; i++) {
  4. Converter<?, String> converter =
  5. converterFactories.get(i).stringConverter(type, annotations, this);
  6. if (converter != null) {
  7. //noinspection unchecked
  8. return (Converter<T, String>) converter;
  9. }
  10. }
  11. return (Converter<T, String>) BuiltInConverters.ToStringConverter.INSTANCE;
  12. }

看到核心代码是converter的stringConverter(type, annotations, this)方法:

因为我们刚才的示例中被没有通过:addConverterFactory(ConverterFactory)添加一个ConverterFactory,所以这里会返回一个空:

  1. public @Nullable Converter<?, String> stringConverter(Type type, Annotation[] annotations,
  2. Retrofit retrofit) {
  3. return null;
  4. }

所以最后会执行最后一句代码:

return (Converter<T, String>) BuiltInConverters.ToStringConverter.INSTANCE;

我们点进去看看这个INSTANCE:

  1. static final ToStringConverter INSTANCE = new ToStringConverter();

是BuiltInConverters内的内部类ToStringConverter的单例。所以这里我们得到的就

是BuiltInConverters.ToStringConverter的实例。

最后用这个对象构建一个Path(因为示例中的参数类型是path,所以我们看这个代码):

  1. new ParameterHandler.Path<>(name, converter, path.encoded());

我们看看这个Path类的构造函数:

  1. Path(String name, Converter<T, String> valueConverter, boolean encoded) {
  2. this.name = checkNotNull(name, "name == null");
  3. this.valueConverter = valueConverter;
  4. this.encoded = encoded;
  5. }

只是赋值,并且我们看到这个类继承自:ParameterHandler,所以我们回到刚才的build()方法,发现把参数类型,参数注解放在一起解析之后存储到了这个ParameterHandler数组中,中间主要做了多种合法性校验,并根据注解的类型,生成不同的

ParameterHandler子类,如注解是Url则生成ParameterHandler.RelativeUrl()对象,如果注解是Path,则生成:

ParameterHandler.Path<>(name, converter, path.encoded())对象等等。

我们查看了ParameterHandler类,发现它有一个抽象方法:

  1. abstract void apply(RequestBuilder builder, @Nullable T value) throws IOException;

这个方法每个子类都必须复写,那我们看看Path里面怎么复写的:

  1. @Override
  2. void apply(RequestBuilder builder, @Nullable T value) throws IOException {
  3. builder.addPathParam(name, valueConverter.convert(value), encoded);
  4. }

就是把value被添加到RequestBuilder中,我们看一下这个addPathParam方法:

  1. void addPathParam(String name, String value, boolean encoded) {
  2. relativeUrl = relativeUrl.replace("{" + name + "}", canonicalizeForPath(value, encoded));
  3. }

这个方法把我们传进来的值value按照编码格式转换,然后替换relativeUrl中的{name},构成一个有效的省略域名的URL。至此,URL的拼接已经完成!

总结:Retrofit使用动态代理模式实现我们定义的网络请求接口,在重写invoke方法的时候构建了一个ServiceMethod对象,在构建这个对象的过程中进行了方法的注解解析得到网络请求方式httpMethod,以及参数的注解分析,拼接成一个省略域名的URL

二.Retrofit网络请求

我们刚才解析了apply方法,我们看看apply方法是谁调用的呢?跟踪一下就发先只有toCall(args);方法:

  1. okhttp3.Call toCall(@Nullable Object... args) throws IOException {
  2. RequestBuilder requestBuilder = new RequestBuilder(httpMethod, baseUrl, relativeUrl, headers,
  3. contentType, hasBody, isFormEncoded, isMultipart);
  4. @SuppressWarnings("unchecked") // It is an error to invoke a method with the wrong arg types.
  5. ParameterHandler<Object>[] handlers = (ParameterHandler<Object>[]) parameterHandlers;
  6. int argumentCount = args != null ? args.length : 0;
  7. if (argumentCount != handlers.length) {
  8. throw new IllegalArgumentException("Argument count (" + argumentCount
  9. + ") doesn't match expected count (" + handlers.length + ")");
  10. }
  11. for (int p = 0; p < argumentCount; p++) {
  12. handlers[p].apply(requestBuilder, args[p]);
  13. }
  14. return callFactory.newCall(requestBuilder.build());
  15. }

这个方法一开始就构建了RequestBuilder,传进去的参数包含:

httpMethod,baseUrl,relativeUrl,headers,contentType,hasBody,isFormEncoded,isMultipart!

然后获取了parameterHandlers,我们上边分析的时候,知道这个数组是存参数注解的解析结果的,并对其进行遍历调用了如下方法:

  1. for (int p = 0; p < argumentCount; p++) {
  2. handlers[p].apply(requestBuilder, args[p]);
  3. }

把参数值传进RequestBuilder中。

最后调用callFactory.newCall(requestBuilder.build())生成一个okhttp3.Call。

我们看一下这个build方法:

  1. Request build() {
  2. HttpUrl url;
  3. HttpUrl.Builder urlBuilder = this.urlBuilder;
  4. if (urlBuilder != null) {
  5. url = urlBuilder.build();
  6. } else {
  7. // No query parameters triggered builder creation, just combine the relative URL and base URL.
  8. //noinspection ConstantConditions Non-null if urlBuilder is null.
  9. url = baseUrl.resolve(relativeUrl);
  10. if (url == null) {
  11. throw new IllegalArgumentException(
  12. "Malformed URL. Base: " + baseUrl + ", Relative: " + relativeUrl);
  13. }
  14. }
  15. RequestBody body = this.body;
  16. if (body == null) {
  17. // Try to pull from one of the builders.
  18. if (formBuilder != null) {
  19. body = formBuilder.build();
  20. } else if (multipartBuilder != null) {
  21. body = multipartBuilder.build();
  22. } else if (hasBody) {
  23. // Body is absent, make an empty body.
  24. body = RequestBody.create(null, new byte[0]);
  25. }
  26. }
  27. MediaType contentType = this.contentType;
  28. if (contentType != null) {
  29. if (body != null) {
  30. body = new ContentTypeOverridingRequestBody(body, contentType);
  31. } else {
  32. requestBuilder.addHeader("Content-Type", contentType.toString());
  33. }
  34. }
  35. return requestBuilder
  36. .url(url)
  37. .method(method, body)
  38. .build();
  39. }

可以看到okhttp的请求体在这里构建,当所有的参数满足的时候,则调用了

  1. Request.Builder requestBuilder
  2. .url(url)
  3. .method(method, body)
  4. .build();

这是发起okhttp的网络请求 。

那这个toCall(args);谁调用的呢?继续往回跟!

  1. private okhttp3.Call createRawCall() throws IOException {
  2. okhttp3.Call call = serviceMethod.toCall(args);
  3. return call;
  4. }

那谁调用了createRawCall()呢?继续看谁调用了!于是发现调用方有三个地方,并且都是OkHttpCall里面!我们一个一个看吧:

  1. Request request()方法:
  2. enqueue(final Callback callback)方法
  3. Response execute()的方法

    很明显上面三个方法都是retrofit的发起网络请求的方式,分别是同步请求和异步请求。我们的示例中在最后一步就是调用了request方法和enqueue方法发起网络请求。至此我们已经疏通了retrofit是如何进行网络请求的了。

    总结:当我们调用Retrofit的网络请求方式的时候,就会调用okhttp的网络请求方式,参数使用的是实现接口的方法的时候拿到的信息构建的RequestBuilder对象,然后在build方法中构建okhttp的Request,最终发起网络请求

三.总结

至此retrofit的流程讲完了,文章很长,代码很多,读者最好下载代码导入IDE,跟着文章一起看代码

Retrofit主要是在create方法中采用动态代理模式实现接口方法,这个过程构建了一个ServiceMethod对象,根据方法注解获取请求方式,参数类型和参数注解拼接请求的链接,当一切都准备好之后会把数据添加到Retrofit的RequestBuilder中。然后当我们主动发起网络请求的时候会调用okhttp发起网络请求,okhttp的配置包括请求方式,URL等在Retrofit的RequestBuilder的build()方法中实现,并发起真正的网络请求。

Retrofit封装了okhttp框架,让我们的网络请求更加简洁,同时也能有更高的扩展性。当然我们只是窥探了Retrofit源码的一部分,他还有更复杂更强大的地方等待我们去探索包括返回值转换工厂,拦截器等,这些都属于比较难的地方,我们需要循序渐进的去学习,当我们一点一点的看透框架的本质之后,我们使用起来才会熟能生巧。大神的代码,对于Android想要进阶的同学来说很有好处,不仅教会我们如何设计代码更多的是解决思想。

Android进阶:七、Retrofit2.0原理解析之最简流程【下】的更多相关文章

  1. Android进阶:七、Retrofit2.0原理解析之最简流程【上】

    retrofit 已经流行很久了,它是Square开源的一款优秀的网络框架,这个框架对okhttp进行了封装,让我们使用okhttp做网路请求更加简单.但是光学会使用只是让我们多了一个技能,学习其源码 ...

  2. Android中微信抢红包插件原理解析和开发实现

    一.前言 自从去年中微信添加抢红包的功能,微信的电商之旅算是正式开始正式火爆起来.但是作为Android开发者来说,我们在抢红包的同时意识到了很多问题,就是手动去抢红包的速度慢了,当然这些有很多原因导 ...

  3. [原理] Android Native内存泄漏检测原理解析

    转载请注明出处:https://www.cnblogs.com/zzcperf/articles/11615655.html 上一篇文章列举了不同版本Android OS内存泄漏的检测操作(传送门), ...

  4. Android进阶(七)数据存储

    Android 数据存储 1访问资源文件 直接将文件保存在设备的内部存储. 默认情况下,保存到内部存储的文件为私有的,其他应用程序不能访问它们,当用户卸载应用程序时,所保存的文件也一并删除.  1.1 ...

  5. Android View 的事件分发原理解析

    作为一名 Android 开发者,每天接触最多的就是 View 了.Android View 虽然不是四大组件,但其并不比四大组件的地位低.而 View 的核心知识点事件分发机制则是不少刚入门同学的拦 ...

  6. 我的Android进阶之旅------>(全解析)屏幕尺寸,分辨率,像素,PPI之间到底什么关系?

    作者:马忠信,作者授权早读课发表,转载请联系作者. 原文链接:http://www.jianshu.com/p/c3387bcc4f6e#  互联网早读课:http://zaodula.com/arc ...

  7. Android进阶系列之源码分析Activity的启动流程

    美女镇楼,辟邪! 源码,是一个程序猿前进路上一个大的而又不得不去翻越障碍,我讨厌源码,看着一大堆.5000多行,要看完得啥时候去了啊.不过做安卓的总有这一天,自从踏上这条不归路,我就认命了.好吧,我慢 ...

  8. APPcrawler基础原理解析及使用

    一.背景 一年前,我们一直在用monkey进行Android 的稳定性测试 ,主要目的就是为了测试app 是否会产生Crash,是否会有ANR,页面错误等问题,在monkey测试过程中,实现了脱离Ca ...

  9. Android DecorView 与 Activity 绑定原理分析

    一年多以前,曾经以为自己对 View 的绘制已经有了解了,事后发现也只是懂了些皮毛而已.经过一年多的实战,Android 和 Java 基础都有了提升,时机成熟了,是时候该去总结 View 的绘制流程 ...

随机推荐

  1. 第二周博客作业<西北师范大学|李晓婷>

    1.助教博客链接:https://home.cnblogs.com/u/lxt-/ 2.点评作业内容: https://www.cnblogs.com/dxd123/p/10494907.html#4 ...

  2. 微信小程序 TLS 版本必须大于等于1.2问题解决

    微信小程序  TLS 版本必须大于等于1.2问题解决 此问题最近在微信小程序开发中,比较常见. 在解决这个问题之前,我们需要了解一下,当前的系统环境是否支持TLS1.2以上,可以参考一下表格: 确认系 ...

  3. python基础学习小结

    Python是一门面向对象的解释性语言(脚本语言),这一类语言的特点就是不用编译,程序在运行的过程中,由对应的解释器向CPU进行翻译,个人理解就是一边编译一边执行.而JAVA这一类语言是需要预先编译的 ...

  4. Ajax简述

    AJAX即“Asynchronous Javascript And XML”(异步JavaScript和XML),是指一种创建交互式网页应用的网页开发技术.AJAX = 异步 JavaScript和X ...

  5. 要求必须全部重复的数据sql--想了半天才写出来的

    CREATE TABLE [dbo].[ABC]( ) NULL, ) NULL, ) NULL, ) NULL ) ON [PRIMARY] GO --DELETE FROM [dbo].[ABC] ...

  6. Asp.net 项目部署的两个问题

    1:关于MVC中BundleCollection压缩js css文件 发布后获取失败的问题 原因是: 默认本地vs里面调试的时候,因为web.config文件里面有一个debug属性,当有此属性时,默 ...

  7. babel-polyfill的几种使用方式

    前言 preset与plugin的关系: preset中已经包含了一组用来转换ES6+的语法的插件,如果只使用少数新特性而非大多数新特性,可以不使用preset而只使用对应的转换插件 babel默认只 ...

  8. 逆元知识普及(扫盲篇) —— from Judge

    watch out 本文是博主的 csdn 上搬过来的,格式有点崩,看不下去的可以去 博主的 csdn上看(上面 格式会好很多,并且有些公式也用 $\LaTeX$  update 上去了) 最近有点颓 ...

  9. [JavaScript]ECMA-6 箭头函数

    概述 箭头函数的作用是为Js提供一种函数的简写方法,箭头函数作用域内不包含this, arguments, super, or new.target,并且不能用于对象的构造函数: 基本语法 [(][p ...

  10. .net mvc的“从客户端中检测到有潜在危险的 Request.Form 值”问题解决

    第一种解决方案 : 在控制器调用的方法上添加[ValidateInput(false)] 第二种解决方案 : 在对应的asp.net web页面上加上ValidateRequest="fal ...