Retrofit

本篇文章只做Retrofit导读,不做细节分析,先来一张Retrofit请求流程图:


Retrofit流程图.png

总结下来就这几点:

1,Retrofit的创建
2,creat创建接口对象
3,调用接口方法生成Call或Observable对象
4,Call调用enqueue或excute方法进行异步或同步请求(Observable的请求是在数据发射源里进行的)
5,对返回结果进行处理

完整的调用流程

public interface MovieService { 

        @GET("top250") 
        Call<MovieEntity> getTopMovie(@Query("start") int start, @Query("count") int count);
        
        @POST("/form")
        @FormUrlEncoded
        Call<ResponseBody> submit(@Field("username") String name, @Field("token") String token);
}

Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(baseUrl)
            .addConverterFactory(GsonConverterFactory.create())
            .build();

 MovieService movieService = retrofit.create(MovieService.class);
    Call<MovieEntity> call = movieService.getTopMovie(0, 10);
    call.enqueue(new Callback<MovieEntity>() {
        @Override
        public void onResponse(Call<MovieEntity> call, Response<MovieEntity> response) {
            resultTV.setText(response.body().toString());
        }

        @Override
        public void onFailure(Call<MovieEntity> call, Throwable t) {
            resultTV.setText(t.getMessage());
        }
    });
}

1,Retrofit的创建

先上菜,在看怎么吃

   Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(baseUrl)
            .addConverterFactory(GsonConverterFactory.create())
            .build();

Retrofit.Builder构造方法
看源码

1.Platform.get() 会拿到一个适合Android平台的对象
2.converterFactories.add(new BuiltInConverters()),converterFactories默认会添加一个BuiltInConverters对象,BuiltInConverters是在添加GsonConverterFactory之前具体可以在后面看BuiltInConverters对数据处理的一些优化

Retrofit.Builder参数设置
参数的设置包含以下属性:

1.okhttp3.Call.Factory 生产网络请求器,用于创建OkHttp3.call对象的工厂类,默认实现类OkHttpClient
2.baseUrl 设置基本baseUrl,通过HttpUrl的parse方法生成符合OkHttp要求的HttpUrl对象(包含一些规则判断,杂物质过滤等)
3.List<Converter.Factory> converterFactories 数据转换工厂类集合,对回调结果进行格式转换
4.List<CallAdapter.Factory> adapterFactories 网络请求适配器工厂的集合
5.Executor callbackExecutor 结果回调器
6.validateEagerly 提前配置接口方法的标志位

Retrofit.Builder的build方法
Retrofit.Builder的build方法创建的Retrofit对象

build方法

build方法过程:

1.如果没有设置okhttp3.Call.Factory,默认创建OkHttpClient,OkHttpClient实现了okhttp3.Call.Factory接口
2.默认通过Platform.get()创建的platform对象的platform.defaultCallbackExecutor()方法创建默认的结果回调器
在Android(继承自Platform)的defaultCallbackExecutor()方法里返回的是MainThreadExecutor回调到主线程
3.通过类似copy的手段把设置的List<CallAdapter.Factory>集成赋值到另一个,以免后续改动影响。也会调用platform的defaultCallAdapterFactory添加一个默认ExecutorCallAdapterFactory(CallAdapter.Factory的实现类)对象
4.数据转换集成的copy
5.创建Retrofit实例
在附上一张Android平台的类源码图:


image.png

2.Retroti的create创建接口对象:

 MovieService movieService = retrofit.create(MovieService.class) creat方法是利用动态代理创建接口对象
creat方法

validateEagerly的作用就是在第一次创建接口对象时配置所有方法的ServiceMethod,通过Map缓存起来

3,调用接口方法生成Call或Observable对象

Call<MovieEntity> call = movieService.getTopMovie(0, 10);

调用getTopMovie方法之后会回调到如上图所示的InvocationHandler对象中的invoke方法,除了通过loadServiceMethod方法生成ServiceMethod对象,还会创建一个OkHttpCall对象,loadServiceMethod


loadServiceMethod

ServiceMethod会调用通过Builder的构造方法传入Retrofit对象和方法method对象,在通过build方法提取所有Retrofit和Method(这里指接口中getTopMovie方法)中的属性,这其中包括对方法上和方法参数中注解的解析
附图两张


ServiceMethod的Builder构造函数
public ServiceMethod build() {
      //创建一个CallAdapter对象最后会调用他的adapt方法创建Call或ObserVable
      callAdapter = createCallAdapter();
      responseType = callAdapter.responseType();
      if (responseType == Response.class || responseType == okhttp3.Response.class) {
        throw methodError("'"
            + Utils.getRawType(responseType).getName()
            + "' is not a valid response body type. Did you mean ResponseBody?");
      }
      responseConverter = createResponseConverter();

      //解析所有的方法注解并赋值
      for (Annotation annotation : methodAnnotations) {
        parseMethodAnnotation(annotation);
      }
      //请求方法不能为空,比如GET,POST必须有一个
      if (httpMethod == null) {
        throw methodError("HTTP method annotation is required (e.g., @GET, @POST, etc.).");
      }
      
      //如果注解是@GET则@Multipart和@FormEncoded都不能存在
      if (!hasBody) {
        if (isMultipart) {
          throw methodError(
              "Multipart can only be specified on HTTP methods with request body (e.g., @POST).");
        }
        if (isFormEncoded) {
          throw methodError("FormUrlEncoded can only be specified on HTTP methods with "
              + "request body (e.g., @POST).");
        }
      }

      //下面是对参数注解的解析
      int parameterCount = parameterAnnotationsArray.length;
      parameterHandlers = new ParameterHandler<?>[parameterCount];
      for (int p = 0; p < parameterCount; p++) {
        Type parameterType = parameterTypes[p];
        if (Utils.hasUnresolvableType(parameterType)) {
          throw parameterError(p, "Parameter type must not include a type variable or wildcard: %s",
              parameterType);
        }

        Annotation[] parameterAnnotations = parameterAnnotationsArray[p];
        if (parameterAnnotations == null) {
          throw parameterError(p, "No Retrofit annotation found.");
        }

        parameterHandlers[p] = parseParameter(p, parameterType, parameterAnnotations);
      }
      
      //没有设置url(在方法注解和参数注解都没有设置url)
      if (relativeUrl == null && !gotUrl) {
        throw methodError("Missing either @%s URL or @Url parameter.", httpMethod);
      }

      //@Body注解的必须同时存在@Multipart或@FormEncoded或POST放等其中一种
      if (!isFormEncoded && !isMultipart && !hasBody && gotBody) {
        throw methodError("Non-body HTTP method cannot contain @Body.");
      }

      //@FormEncoded注解至少要一个参数注解是@Field
      if (isFormEncoded && !gotField) {
        throw methodError("Form-encoded method must contain at least one @Field.");
      }
      
       //@Multipart注解至少要一个参数注解是@Part
      if (isMultipart && !gotPart) {
        throw methodError("Multipart method must contain at least one @Part.");
      }

      return new ServiceMethod<>(this);
    }

serviceMethod.callAdapter.adapt(okHttpCall)中的 serviceMethod.callAdapter是在build方法中创建的,如果是默认的ExecutorCallAdapterFactory的get方法创建的callAdapter(ExecutorCallAdapterFactory 中的ExecutorCallbackCall)则调用adapt方法是会返回Call对象,如果是设置了RxJavaCallAdapterFactory则通过get方法创建的callAdapter对象调用adapt方法会返回Observable对象

4,发送网络请求

 call.enqueue(new Callback<MovieEntity>() {
        @Override
        public void onResponse(Call<MovieEntity> call, Response<MovieEntity> response) {
            resultTV.setText(response.body().toString());
        }

        @Override
        public void onFailure(Call<MovieEntity> call, Throwable t) {
            resultTV.setText(t.getMessage());
        }
    });

通过serviceMethod.callAdapter.adapt(okHttpCall)创建的Call对象调用enqueue或excute方法进行异步或同步请求,Call实则是OkHttpCall的代理类,调用的其实的OkHttpCall的enqueue或excute方法,OkHttpCall的excute方法会创建OkHttp的call对象进行网络请求,如图


OkHttp的enqueque方法

Observable的网络请求实则是在数据发射源里进行的,具体看RxJavaCallAdapterFactory

5,回调结果的处理

CallAdapter(ExecutorCallbackCall实现类)中会通过callbackExecutor将结果发送到主线程


ExecutorCallbackCall

callbackExecutor的默认实现类是MainThreadExecutor,Rx可以自由切换线程

最后盗一张图


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

推荐阅读更多精彩内容