Retrofit梳理

2,700 阅读5分钟

本文为过往笔记整理, 在此只作记录,不做严谨的技术分享。

概述

RESTful

理解RESTful架构--阮一峰
RESTful API 最佳实践--阮一峰

Representational State Transfer ,架构风格(不是标准),使用正确的格式来请求和响应

  • 使用资源格式来定义URL
  • 规范使用method来定义网络请求操作
    • GET、POST、DELETE等方法,后台定义好的规则前端遵循。
    • 但现在存在的问题是,后台并不一定会按照规则设计接口。
  • 规范使用status code来表示响应状态
    • statuscode、message需按规则设计返回的内容。

Retrofit

Retrofit本质是一个RESTful风格的网络请求框架,用Java接口来描述Http请求,在接口里通过注解和动态代理,生成Request由OKHttp进行真正的网络请求

注解

Retrofit 共22个注解,根据功能大概分为三类

分类注解
方法注解@GET、@POST、@HEAD、@PUT、@DELETE、@HTTP、@PATH、@OPTIONS
标记注解@FormUrlEncoded、@Multipart、@Streaming
参数注解@Query、@QueryMap、@Body、@Field、@FieldMap、@Part、@PartMap、@Path、@Url、@Header、@Headers

方法注解

@GET、@POST、@HEAD、@PUT、@DELETE

分别对应 HTTP中的网络请求方式

@HTTP

替换@GET、@POST、@PUT、@DELETE、@HEAD注解的作用, 及更多功能拓展
具体使用:通过属性method、path、hasBody进行设置

/**
 * method:网络请求的方法(区分大小写)
 * path:网络请求地址路径
 * hasBody:是否有请求体
 */
@HTTP(method = "GET", path = "blog/{id}", hasBody = false)
Call<ResponseBody> getCall(@Path("id") int id);

标记注解

@FormUrlEncoded、@Multipart、@Streaming

@FormUrlEncoded

表示发送form-encoded的数据,每个键值对需要用@Filed来注解键名,随后的对象需要提供值。

/**
  *表明是一个表单格式的请求(Content-Type:application/x-www-form-urlencoded)
  * <code>Field("username")</code> 表示将后面的 <code>String name</code>    中name的取值作为 username 的值
  */
 @POST("/form")
 @FormUrlEncoded
 Call<ResponseBody> testFormUrlEncoded1(@Field("username") String name,@Field("age") int age);
@Multipart

表示发送form-encoded的数据(适用于 有文件 上传的场景),每个键值对需要用@Part来注解键名,随后的对象需要提供值。

/**
 * {@link Part} 后面支持三种类型,{@link RequestBody}、{@link    okhttp3.MultipartBody.Part} 、任意类型
 * 除 {@link okhttp3.MultipartBody.Part}    以外,其它类型都必须带上表单字段({@link okhttp3.MultipartBody.Part}    中已经包含了表单字段的信息),
 */
@POST("/form")
@Multipart
Call<ResponseBody> testFileUpload1(@Part("name") RequestBodyname,@Part("age")    RequestBody age, @Part MultipartBody.Part file);

参数注解

@Body、@Path、@Url
@Field、@FieldMap
@Query、@QueryMap
@Part、@PartMap
@Header、@Headers

@Body

以 Post方式传递,自定义数据类型给服务器
如果提交的是一个Map,那么作用相当于 @Field,不过Map要经过 FormBody.Builder类处理成为符合 Okhttp格式的表单,如:

FormBody.Builder builder = new FormBody.Builder();
builder.add("key","value");
FormBody body = builder.build();

HashMap<String, String> map = new HashMap<>();
RequestBody body = RequestBody
    .create(MediaType.parse("application/json;charset=utf-8"), new Gson().toJson(map));

Map body = new HashMap();

@POST("api/gameKkz/receiveGameKkzBox")
Observable<BaseResponse<Empty>> test(@Body RequestBody body);
Observable<BaseResponse<Empty>> test(@Body Map<String,Object> map);
@Path

URL地址的缺省值

/**
* 访问的API是:https://api.github.com/users/{user}/repos
* 在发起请求时, {user} 会被替换为方法的第一个参数user(被@Path注解作用)
*/
@GET("users/{user}/repos")
Call<ResponseBody>  getBlog(@Path("user") String user );
@Url

直接传入一个请求的 URL变量 用于URL设置

/**
* 当有URL注解时,@GET传入的URL就可以省略
* 当GET、POST...HTTP等方法中没有设置Url时,则必须使用 {@link Url}提供
*/
@GET
Call<ResponseBody> testUrlAndQuery(@Url String url, @Query("showAll") boolean showAll);
@Field & @FieldMap

发送 Post请求时提交请求的表单字段,与 @FormUrlEncoded 注解配合使用

/**
 *表明是一个表单格式的请求(Content-Type:application/x-www-form-urlencoded)
 * <code>Field("username")</code> 表示将后面的 <code>String name</code> 中name的取值作为 username 的值
 */
@POST("/form")
@FormUrlEncoded
Call<ResponseBody> testFormUrlEncoded1(@Field("username") String name, @Field("age") int age);

/**
 * Map的key作为表单的键
 */
@POST("/form")
@FormUrlEncoded
Call<ResponseBody> testFormUrlEncoded2(@FieldMap Map<String, Object>map);
@Query和@QueryMap

用于 @GET 方法的查询参数(Query = Url 中 ‘?’ 后面的 key-value)
如:url = www.println.net/?cate=andro… = cate 配置时只需要在接口方法中增加一个参数即可:

// 其使用方式同 @Field与@FieldMap
@GET("/")    
Call<String> cate(@Query("cate") String cate);
@Part & @PartMap

发送 Post请求时提交请求的表单字段
与@Field的区别:功能相同,但携带的参数类型更加丰富,包括数据流,所以适用于 有文件上传 的场景,与 @Multipart 注解配合使用

 /**
 * {@link Part} 后面支持三种类型,{@link RequestBody}、{@link okhttp3.MultipartBody.Part} 、任意类型
 * 除 {@link okhttp3.MultipartBody.Part} 以外,其它类型都必须带上表单字段({@link okhttp3.MultipartBody.Part} 中已经包含了表单字段的信息),
 */
@POST("/form")
@Multipart
Call<ResponseBody> testFileUpload1(@Part("name") RequestBody name, @Part("age") RequestBody age, @Part MultipartBody.Part file);

/**
 * PartMap 注解支持一个Map作为参数,支持 {@link RequestBody } 类型,
 * 如果有其它的类型,会被{@link retrofit2.Converter}转换,如后面会介绍的 使用{@link com.google.gson.Gson} 的 {@link retrofit2.converter.gson.GsonRequestBodyConverter}
 * 所以{@link MultipartBody.Part} 就不适用了,所以文件只能用<b> @Part MultipartBody.Part </b>
 */
@POST("/form")
@Multipart
Call<ResponseBody> testFileUpload2(@PartMap Map<String, RequestBody> args, @Part MultipartBody.Part file);

@POST("/form")
@Multipart
Call<ResponseBody> testFileUpload3(@PartMap Map<String, RequestBody>args);
@Header & @Headers

效果是一致的,区别在于使用场景和使用方式:
使用场景:@Header用于添加不固定的请求头,@Headers用于添加固定的请求
使用方式:@Header作用于方法的参数,@Headers作用于方法

// @Header
@GET("user")
Call<User> getUser(@Header("Authorization") String authorization)

// @Headers
@Headers("Authorization: authorization")
@GET("user")
Call<User> getUser()

使用

Retrofit retrofit = new Retrofit.Builder()
        .baseUrl(ApiRetrofit.BASE_URL) 
        //都可以添加多个Factory
        .addConverterFactory(GsonConverterFactory.create())
        //.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
        .build();        
ApiRetrofit apiRetrofit = retrofit.create(ApiRetrofit.class);

Call<Result<List<Chapter>>> call = apiRetrofit.chapters();
call.enqueue(new Callback<Result<List<Chapter>>>() {
    @Override
    public void onResponse(Call<Result<List<Chapter>>> call, 
                           Response<Result<List<Chapter>>> response) {
        Result<List<Chapter>> list = response.body();
    }

    @Override
    public void onFailure(Call<Result<List<Chapter>>> call, Throwable t) {
    }
});

原理

使用动态代理,根据method上的注解封装请求数据

  public <T> T createService(Class<T> clz) {
    return retrofit.create(clz);
  }

  public <T> T create(final Class<T> service) {
    validateServiceInterface(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 @Nullable 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);
          }
        });
  }

  //根据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;
  }

设计模式

  • 工厂模式
  • 适配模式
  • Builder
  • 策略模式
  • 代理模式