Retrofit(一)

Android知识总结

一、简介

1、Retrofit是什么?

准确来说,Retrofit 是一个 RESTful 的 HTTP 网络请求框架的封装。
原因:网络请求的工作本质上是 OkHttp 完成,而 Retrofit 仅负责 网络请求接口的封装

我们先来看看下面这个图:


上图说明了如下几点:

    1. App应用程序通过 Retrofit 请求网络,实际上是使用 Retrofit 接口层封装请求参数、Header、Url 等信息,之后由 OkHttp 完成后续的请求操作。
    1. 在服务端返回数据之后,OkHttp 将原始的结果交给 Retrofit,Retrofit根据用户的需求对结果进行解析。

所以,网络请求的本质仍旧是OkHttp完成的,retrofit只是帮使用者来进行工作简化的,比如配置网络,处理数据等工作,提高这一系列操作的复用性。这也就是网上流行的一个不太准确的总结:okhttp是瑞士军刀,retrofit则是讲这边瑞士军刀包装从了一个非常好用的指甲钳。

2、Retrofit用到的 8 大设计模式

  • 1)Retrofit 实例使用建造者模式通过Builder类构建。
    当构造函数的参数大于4个,
    且存在可选参数的时候既可以使用 建造者设计模式

  • 2)Retrofit 创建时的callFactory,使用工厂方法设计模式,但是似乎并不打算支持其他的工厂。

  • 3)整个retrofit 采用的时外观模式。统一的调用创建网络请求接口实例和网络请求参数配置的方法。

  • 4)retrofit里面使用了动态代理来创建网络请求接口实例,这个是retrofit对用户使用来说最大的复用,其它的代码都是为了支撑这个动态代理给用户带来便捷性的。

  • 5)使用了策略模式对serviceMethod对象进行网络请求参数配置,即通过解析网络请求接口方法的参数、返回值和注解类型,从Retrofit对象中获取对应的网络的url地址、网络请求执行器、网络请求适配器和数据转换器。

  • 6)ExecuteCallBack 使用装饰者模式来封装callbackExecutor,用于完成线程的切换。

  • 7)ExecutorCallbackCall 使用静态代理(委托) 代理了Call进行网络请求,真正的网络请求由okhttpCall执行,然而okHttpCall不是自己执行,它是okhttp 提供call给 外界(retrofit)使用的唯一门户,其实这个地方就是门面模式

  • 8)ExecutorCallbackCall 的被初始化是在ExecutorCallAdapterFactory里面通过适配器模式被创建的。
    CallAdapter采用了适配器模式 为创建访问Call接口提供服务。默认不添加Rxjava则使用默认的ExecutorCallAdapterFactory 将okhttp3.call转变成为 retroift中的call,如果有Rxjava则将okhttp3.call转化为abservable。

3、依赖库

    implementation  'com.squareup.retrofit2:retrofit:2.3.0'
    implementation 'com.squareup.retrofit2:adapter-rxjava2:2.3.0'
    implementation 'com.squareup.retrofit2:converter-gson:2.3.0'

4、注解类型

二、请求方法类

2.1、序号 1 ~ 7

  • 分别对应 HTTP 的请求方法;
  • 接收一个字符串表示接口 path ,与 baseUrl 组成完整的 Url;
  • 可以不指定,结合 @Url 注解使用;
  • url 中可以使用变量,如 {id} ,并使用 @Path(“id”) 注解为 {id} 提供值
@GET("project/tree/json")
Call<ProjectBean> getProject1();
//简单的get请求(URL中带有两个参数)
  @GET("news/{userId}")
  Call<TradesBean> getItem(@Path("userId") String userId,@Path("type") String type);

2.2、序号 8

  • 可用于替代以上 7 个,及其他扩展方法;
  • 有 3 个属性:method、path、hasBody、 举个例子
@HTTP(method = "get", path = "project/tree/json",hasBody = false)
 Call<ProjectBean> getProject2();

三、标记类

3.1、FormUrlEncoded

登录页面使用: Content-Type:application/x-www-form-urlencoded

  • 用于修饰Field注解和FieldMap注解
  • 使用该注解,表示请求正文将使用表单网址编码。字段应该声明为参数,并用@Field注释或FieldMap注释。使用FormUrlEncoded注解的请求将具application/ x-www-form-urlencodedMIME类型。字段名称和值将先进行UTF-8进行编码,再根据RFC-3986进行URI编码.
//普通参数
@FormUrlEncoded
@POST("/")
Call<ResponseBody> example(@Field("name") String name,@Field("occupation") String occupation);

//固定或可变数组
@FormUrlEncoded
@POST("/list")
Call<ResponseBody> example(@Field("name") String... names);

3.1、Multipart

上传文件使用: Content-Type:multipart/form-data
使用该注解,表示请求体是多部分的,每个部分作为一个参数,且用PartPartMap注解声明。

  • 上传单张图片
///////上传单张图片//////
  /**
  * Multipart:表示请求实体是一个支持文件上传的Form表单,需要配合使用@Part,适用于
有文件 上传的场景
  * Part:用于表单字段,Part和PartMap与Multipart注解结合使用,适合文件上传的情况
  * PartMap:用于表单字段,默认接受的类型是Map<String,REquestBody>,可用于实现多
文件上传
  * Part 后面支持三种类型,{@link RequestBody}、{@link
okhttp3.MultipartBody.Part} 、任意类型;
  *
  * @param file 服务器指定的上传图片的key值
  * @return
  */
@Multipart
@POST("project/upload")
Call<ProjectBean> upload1(@Part("file" + "\";filename=\"" + "test.png") RequestBody file);

@Multipart
@POST("project/xxx")
Call<ProjectBean> upload2(@Part MultipartBody.Part file);

////////请求///////////
//上传单个图片1
File file = new File("");
RequestBody requestBody = RequestBody.create(MediaType.parse("image/png"), file);
wanAndroidApi.upload1(requestBody).execute();

//上传单个图片2
MultipartBody.Part imagePart = MultipartBody.Part.createFormData("上传的key", file.getName(), requestBody);
wanAndroidApi.upload2(imagePart)
      .enqueue(new Callback<ProjectBean>() {
      @Override
      public void onResponse(Call<ProjectBean> call, Response<ProjectBean> response) { }
        @Override
        public void onFailure(Call<ProjectBean> call, Throwablet) { }
       });
  • 上传多张图片
///////上传多张图片//////
@Multipart
@POST("project/upload")
Call<ProjectBean> upload3(@PartMap Map<String, RequestBody> map);

@Multipart
@POST("project/xxx")
Call<ProjectBean> upload4(@PartMap Map<String, MultipartBody.Part> map);

////////使用//////////
//上传多张图片1
//图片集合
List<File> files = new ArrayList<>();
Map<String, RequestBody> map = new HashMap<>();
for (int i = 0; i < files.size(); i++) {
   RequestBody requestBody = RequestBody.create(MediaType.parse("image/png"), files.get(i));
   map.put("file" + i + "\";filename=\"" + files.get(i).getName(), requestBody);
}
wanAndroidApi.upload3(map).execute();


//上传多张图片2
Map<String, MultipartBody.Part> map1 = new HashMap<>();

File file1 = new File("");
RequestBody requestBody1 = RequestBody.create(MediaType.parse("image/png"), file1);
MultipartBody.Part part1 = MultipartBody.Part.createFormData("上传的key1", file1.getName(), requestBody1);
map1.put("上传的key1", part1);

File file2 = new File("");
RequestBody requestBody2 = RequestBody.create(MediaType.parse("image/png"), file2);
MultipartBody.Part part2 = MultipartBody.Part.createFormData("上传的key2", file2.getName(), requestBody2);
map1.put("上传的key2", part2);

wanAndroidApi.upload4(map1).execute();
  • 图文混传
//////图文混传/////
  /**
  * @param params
  * @param files
  * @return
  */
  @Multipart
  @POST("upload/upload")
  Call<ProjectBean> upload5(@FieldMap() Map<String, String> params, @PartMap() Map<String, RequestBody> files);

  /**
  * Part 后面支持三种类型,{@link RequestBody}、{@link
okhttp3.MultipartBody.Part} 、任意类型;
  *
  * @param userName
  * @param passWord
  * @param file
  * @return
  */
  @Multipart
  @POST("project/xxx")
  Call<ProjectBean> upload6(@Part("username") RequestBody userName,
               @Part("password") RequestBody passWord,
               @Part MultipartBody.Part file);


    //////使用///////
    MediaType textType = MediaType.parse("text/plain");
    RequestBody name = RequestBody.create(textType, "zero");
    RequestBody password = RequestBody.create(textType, "123456");
    File file = new File("");
    RequestBody requestBody = RequestBody.create(MediaType.parse("image/png"), file);
    MultipartBody.Part part = MultipartBody.Part.createFormData("上传的key", file.getName(), requestBody);

    wanAndroidApi
       .upload6(name, password, part)
       .enqueue(new Callback<ProjectBean>() {
          @Override
          public void onResponse(Call<ProjectBean> call, Response<ProjectBean> response) {
         }
          @Override
          public void onFailure(Call<ProjectBean> call, Throwable t) {
         }
       });

3.1、Streaming

未使用@Straming 注解,默认会把数据全部载入内存,之后通过流获取数据也是读取内存中数据,所以返回数据较大时,需要使用该注解。
处理返回Response的方法的响应体,用于下载大文件。

/**
  * Streaming注解:表示响应体的数据用流的方式返回,
  * 适用于返回的数据比较大,该注解在在下载大文件的特别有用
  */
  @Streaming
  @GET
  Call<ProjectBean> downloadFile(@Url String fileUrl);

四、参数注解

注意:

  • 1、Map用来组合复杂的参数,并且对于FieldMap,HeaderMap,PartMap,QueryMap这四种作用方法的注解,其参数类型必须为Map实例,且key的类型必须为String类型,否则抛出异常。
  • 2、Query、QueryMap与Field、FieldMap功能一样,生成的数据形式一样;Query、QueryMap的数据体现在Url上;Field、FieldMap的数据是请求体
  • 3、{占位符}和PATH尽量只用在URL的path部分,url的参数使用Query、QueryMap代替,保证接口的简洁
  • 4、Query、Field、Part支持数据和实现了iterable接口的类型,如List、Set等,方便向后台传递数组,代码如下:
  • 5、以上部分注解真正的实现在ParameterHandler类中,每个注解的真正实现都是ParameterHandler类中的一个final类型的内部类,每个内部类都对各个注解的使用要求做了限制,比如参数是否可空、键和值是否可空等。
  • 6、@FormUrlEncoded 注解和@Multipart 注解不能同时使用,否则会抛出methodError(“Only one encoding annotation is allowed.”),可在ServiceMethod类中parseMethodAnnotation()方法中找到不能同时使用的具体原因。
  • 7、@Path 与@Url 注解不能同时使用,否则会抛出parameterError(p, "@Path parameters may not be used with @Url."),可在ServcieMethod类中parseParameterAnnotation()方法中找到不能同时使用的具体代码。其实原因也是很好理解:Path注解用于替换url中的参数,这就要求在使用path注解时,必须已经存在请求路径。不然没法替换路径中指定的参数。而@Url 注解是在参数中指定了请求路径的,这时候情定请求路径已经晚,path注解找不到请求路径,更别提更换请求路径了中的参数了。
  • 8 使用@Body 注解的参数不能使用form 或multi-part编码,即如果为方法使用了FormUrlEncodedMultipart注解,则方法的参数中不能使用@Body 注解,否则会抛出异常parameterError(p, “@Body parameters cannot be used with form or multi-part encoding.”)

4.1、Headers

使用 @Headers 注解设置固定的请求头,所有请求头不会相互覆盖,即使名字相同

@Headers("Cache-Control: max-age=640000")
@GET("project/list")
Call<ProjectBean> getMsg1();

@Headers({ "Accept: application/vnd.github.v3.full+json","User-Agent:
Retrofit-Sample-App"})
@GET("project/{username}")
Call<ProjectBean> getMsg2(@Path("username") String username);

4.2、Header

使用 @Header 注解动态更新请求头,匹配的参数必须提供给 @Header ,若参数值为 null ,这个头会被省略,否则,会使用参数值的 toString 方法的返回值。
具有相同名称的请求头不会相互覆盖,而是照样添加到请求头中。

@GET("project")
Call<ProjectBean> getProject3(@Header("Authorization") String authorization);

4.3、Body

使用 @Body 注解,指定一个对象作为 request body,使用@Body 注解定义的参数不能为null。
当你发送一个post或put请求,但是又不想作为请求参数或表单的方式发送请求时,使用该注解定义的参数可以直接传入一个实体类,retrofit会通过convert把该实体序列化并将序列化的结果直接作为请求体发送出去。

   public class ProjectBean {
    private String owner;
    private String name;
    ProjectBean (String owner, String name) {
      this.owner = owner;
      this.name = name;
    }
  }

@POST("project/new")
Call<ProjectBean> createProject(@Body ProjectBean user);

4.4、Field

  • 作用于方法的参数
  • 用于发送一个表单请求
  • 用String.valueOf()把参数值转换为String,然后进行URL编码,当参数值为null值时,会自动忽略,如果传入的是一个List或array,则为每一个非空的item拼接一个键值对,每一个键值对中的键是相同的,值就是非空item的值,如:name=张三&name=李四&name=王五,另外,如果item的值有空格,在拼接时会自动忽略,例如某个item的值为:张三,则拼接后为name=张三。
//固定或可变数组
@FormUrlEncoded
@POST("/list")
Call<ResponseBody> example(@Field("name") String... names);

4.5、FieldMap

  • 作用于方法的参数
  • 用于发送一个表单请求
  • map中每一项的键和值都不能为空,否则抛出IllegalArgumentException异常
FormUrlEncoded
@POST("/examples")
Call<ResponseBody> example(@FieldMap Map<String, String> fields);

4.6、Part

  • 作用于方法的参数,用于定义Multipart请求的每个part
  • 使用该注解定义的参数,参数值可以为空,为空时,则忽略
  • 使用该注解定义的参数类型有以下3种方式可选:
    1、如果类型是okhttp3.MultipartBody.Part,内容将被直接使用。省略part中的名称,即 @Part MultipartBody.Part part
    2、如果类型是RequestBody,那么该值将直接与其内容类型一起使用。在注释中提供part名称(例如,@Part(“foo”)RequestBody foo)
    3、其他对象类型将通过使用转换器转换为适当的格式。 在注释中提供part名称(例如,@Part(“foo”)Image photo
@Multipart
@POST("/")
Call<ResponseBody> example(
       @Part("description") String description,
       @Part(value = "image", encoding = "8-bit") RequestBody image);

//上传一张图片
@Multipart
@POST("you methd url upload/")
Call<ResponseBody> uploadFile(@Part("avatar\\\\"; filename=\\\\"avatar.jpg") RequestBody file);

//上传图片数量不定
@Multipart
@POST("{url}")
Observable<ResponseBody> uploadFiles(
        @Path("url") String url,
        @Part("filename") String description,
        @PartMap() Map<String, RequestBody> maps);

//上传图片数量不定方法2
@Multipart
@POST("{url}")
Observable<ResponseBody> uploads(
        @Path("url") String url,
        @Part("description") RequestBody description,
        @Part("filekey") MultipartBody.Part file);

4.7、PartMap

  • 作用于方法的参数,以map的方式定义Multipart请求的每个part
  • map中每一项的键和值都不能为空,否则抛出IllegalArgumentException异常
  • 使用该注解定义的参数类型有以下2种方式可选:
    1、如果类型是RequestBody,那么该值将直接与其内容类型一起使用
    2、其他对象类型将通过使用转换器转换为适当的格式
@Multipart
@POST("upload/upload")
Call<ProjectBean> upload5(@FieldMap() Map<String, String> params,
               @PartMap() Map<String, RequestBody> files,
               @Part("file") RequestBody file,
               @PartMap Map<String,RequestBody> maps)

4.8、HeaderMap

  • 作用于方法的参数,用于添加请求头
  • 以map的方式添加多个请求头,map中的key为请求头的名称,value为请求头的值,且value使用 String.valueOf() 统一转换为String类型,
  • map中每一项的键和值都不能为空,否则抛出IllegalArgumentException异常
@GET("/example1")
Call<ProjectBean> example1(@HeaderMap Map<String, String> headers);

/////使用///////
Map<String,String> headers = new HashMap<>();
headers.put("Accept","text/plain");
headers.put("Accept-Charset", "utf-8");
wanAndroidApi.example1(headers)
       .enqueue(new Callback<ProjectBean>() {
          @Override
          public void onResponse(Call<ProjectBean> call, Response<ProjectBean> response) { }
          @Override
          public void onFailure(Call<ProjectBean> call, Throwablet) { }
       });

4.9、Path

请求 URL 可以替换模块来动态改变,替换模块是 {}包含的字母数字字符串,替换的参数必须使用 @Path 注解的相同字符串

@GET("example5/{id}")
Call<ResponseBody> example5(@Path("id") int id);

4.10、Query

  • 作用于方法的参数
  • 用于添加查询参数,即请求参数
  • 参数值通过String.valueOf()转换为String并进行URL编码
  • 使用该注解定义的参数,参数值可以为空,为空时,忽略该值,当传入一个List或array时,为每个非空item拼接请求键值对,所有的键是统一的,如:name=张三&name=李四&name=王五
@GET("/list")
Call<ResponseBody> list(@Query("page") int page);
@GET("/list")
Call<ResponseBody> list(@Query("category") String category);
//传入一个数组
@GET("/list")
Call<ResponseBody> list(@Query("category") String... categories);
//不进行URL编码
@GET("/search")
Call<ResponseBody> list(@Query(value="foo", encoded=true) String foo);

4.11、QueryMap

复杂的查询参数

//使用默认URL编码
@GET("/search")
Call<ResponseBody> list(@QueryMap Map<String, String> filters);
//不使用默认URL编码
@GET("/search")
Call<ResponseBody> list(@QueryMap(encoded=true) Map<String, String> filters);

4.12、Url

  • 作用于方法参数
  • 用于添加请求的接口地址
@GET
Call<ResponseBody> example4(@Url String url);

五、Retrfit文件上传

1.创建描述网络请求的接口

 public interface FileUploadService {  
 @Multipart //表示发送form-encoded的数据(适用于 有文件 上传的场景)
 @POST("upload")
 Call<ResponseBody> upload(@Part("description") RequestBody description,
                          @Part MultipartBody.Part file);
}

2.具体使用

// 创建 RequestBody,用于封装 请求RequestBody
RequestBody requestFile =
        RequestBody.create(MediaType.parse("multipart/form-data"), file);

// MultipartBody.Part  和后端约定好Key,这里的partName是用image
MultipartBody.Part body =
        MultipartBody.Part.createFormData("image", file.getName(), requestFile);

// 添加描述
String descriptionString = "hello, 这是文件描述";
RequestBody description =
        RequestBody.create(
                MediaType.parse("multipart/form-data"), descriptionString);

// 执行请求
Call<ResponseBody> call = service.upload(description, body);
call.enqueue(new Callback<ResponseBody>() {
    @Override
    public void onResponse(Call<ResponseBody> call,
                           Response<ResponseBody> response) {
        Log.v("Upload", "success");
    }

    @Override
    public void onFailure(Call<ResponseBody> call, Throwable t) {
        Log.e("Upload error:", t.getMessage());
    }
});

    /**
     * 文字参数
     * @param value
     * @return
     */
    public static RequestBody toRequestBodyOfText (String value) {
        RequestBody body = RequestBody.create(MediaType.parse("text/plain"), value);
        return body ;
    }

    /**
     * 图片文件参数
     * @param pFile
     * @return
     */
    public static RequestBody toRequestBodyOfImage(File pFile){
        RequestBody fileBody = RequestBody.create(MediaType.parse("multipart/form-data"), pFile);
        return fileBody;
    }

    /**
     * 加工图片参数
     * @param params
     * @param file 图片路径
     * @param key 和后台约定
     * @return
     */
    public static HashMap<String,RequestBody> toPutImgFileParams(HashMap<String,RequestBody> params,String key,File file){
        if (params == null) {
            throw new RuntimeException("params is not allow null");
        }

        if (key == null) {
            throw new RuntimeException("file key is not allow null");
        }

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