Retrofit各个注解的含义及作用

写在前面

本篇文章基于retrofit-2.1进行分析.

1. 各个注解的含义及使用

1.1 Body注解:

作用于方法的参数

使用该注解定义的参数不可为null

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

示例:

//实体class Repo {finalString owner;finalString name;    Repo(String owner, String name) {this.owner = owner;this.name = name;    }  }//接口interface Service {@POST("/")    Call sendNormal(@BodyRepo repo);

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

1.2 DELETE注解:

用于发送一个DELETE请求

DELETE注解一般必须添加相对路径或绝对路径或者全路径,如果不想在DELETE注解后添加请求路径,则可以在方法的第一个参数中用@Url注解添加请求路径

1.3 Field注解:

作用于方法的参数

用于发送一个表单请求

用String.valueOf()把参数值转换为String,然后进行URL编码,当参数值为null值时,会自动忽略,如果传入的是一个List或array,则为每一个非空的item拼接一个键值对,每一个键值对中的键是相同的,值就是非空item的值,如: name=张三&name=李四&name=王五,另外,如果item的值有空格,在拼接时会自动忽略,例如某个item的值为:张 三,则拼接后为name=张三.

示例:

//普通参数@FormUrlEncoded@POST("/")Call example(@Field("name") String name,@Field("occupation") String occupation);//固定或可变数组@FormUrlEncoded@POST("/list")  Call example(@Field("name") String... names);

1

2

3

4

5

6

7

8

9

1

2

3

4

5

6

7

8

9

1.4 FieldMap注解:

作用于方法的参数

用于发送一个表单请求

map中每一项的键和值都不能为空,否则抛出IllegalArgumentException异常

示例:

@FormUrlEncoded@POST("/things")  Call things(@FieldMapMap fields);

1

2

3

1

2

3

1.5 FormUrlEncoded注解:

用于修饰Field注解和FieldMap注解

使用该注解,表示请求正文将使用表单网址编码。字段应该声明为参数,并用@Field注释或FieldMap注释。使用FormUrlEncoded注解的请求将具”application / x-www-form-urlencoded” MIME类型。字段名称和值将先进行UTF-8进行编码,再根据RFC-3986进行URI编码.

1.6 GET注解

用于发送一个get请求

GET注解一般必须添加相对路径或绝对路径或者全路径,如果不想在GET注解后添加请求路径,则可以在方法的第一个参数中用@Url注解添加请求路径

1.7 HEAD注解

用于发送一个HEAD请求

HEAD注解一般必须添加相对路径或绝对路径或者全路径,如果不想在HEAD注解后添加请求路径,则可以在方法的第一个参数中用@Url注解添加请求路径

1.8 Header注解:

作用于方法的参数,用于添加请求头

使用该注解定义的请求头可以为空,当为空时,会自动忽略,当传入一个List或array时,为拼接每个非空的item的值到请求头中.

具有相同名称的请求头不会相互覆盖,而是会照样添加到请求头中

示例:

@GET("/")  Call foo(@Header("Accept-Language") String lang);

1

2

1

2

1.9 HeaderMap注解:

作用于方法的参数,用于添加请求头

以map的方式添加多个请求头,map中的key为请求头的名称,value为请求头的值,且value使用String.valueOf()统一转换为String类型,

map中每一项的键和值都不能为空,否则抛出IllegalArgumentException异常

示例:

@GET("/search")voidlist(@HeaderMapMap headers);//mapMap headers =newHashMap()<>;  headers.put("Accept","text/plain");  headers.put("Accept-Charset","utf-8");

1

2

3

4

5

6

7

1

2

3

4

5

6

7

2.0 Headers注解:

作用于方法,用于添加一个或多个请求头

具有相同名称的请求头不会相互覆盖,而是会照样添加到请求头中

示例:

//添加一个请求头@Headers("Cache-Control: max-age=640000")@GET("/") ...//添加多个请求头@Headers({"X-Foo: Bar","X-Ping: Pong"})@GET("/")  ...

1

2

3

4

5

6

7

8

9

10

1

2

3

4

5

6

7

8

9

10

2.1 HTTP注解:

作用于方法,用于发送一个自定义的HTTP请求

示例:

//自定义HTTP请求的标准样式interface Service {@HTTP(method ="CUSTOM", path ="custom/endpoint/")    Call customEndpoint();  }//发送一个DELETE请求interface Service {@HTTP(method ="DELETE", path ="remove/", hasBody =true)    Call deleteObject(@BodyRequestBody object);  }

1

2

3

4

5

6

7

8

9

10

1

2

3

4

5

6

7

8

9

10

2.2 Multipart注解:

作用于方法

使用该注解,表示请求体是多部分的。 每一部分作为一个参数,且用Part注解声明

2.3 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 example(@Part("description") String description,@Part(value ="image", encoding ="8-bit") RequestBody image);

1

2

3

4

5

1

2

3

4

5

2.4 PartMap注解:

作用于方法的参数,以map的方式定义Multipart请求的每个part

map中每一项的键和值都不能为空,否则抛出IllegalArgumentException异常

使用该注解定义的参数类型有以下2种方式可选:

1,如果类型是RequestBody,那么该值将直接与其内容类型一起使用。

2,其他对象类型将通过使用转换器转换为适当的格式。

示例:

@Multipart@POST("/upload")Call upload(@Part("file") RequestBody file,@PartMapMap params);

1

2

3

4

5

1

2

3

4

5

2.5 OPTIONS注解:

用于发送一个OPTIONS请求

OPTIONS注解一般必须添加相对路径或绝对路径或者全路径,如果不想在OPTIONS注解后添加请求路径,则可以在方法的第一个参数中用@Url注解添加请求路径

2.6 PATCH注解:

用于发送一个PATCH请求

PATCH注解一般必须添加相对路径或绝对路径或者全路径,如果不想在PATCH注解后添加请求路径,则可以在方法的第一个参数中用@Url注解添加请求路径

2.7 Path注解:

作用于方法的参数

在URL路径段中替换指定的参数值。使用String.valueOf()和URL编码将值转换为字符串。

使用该注解定义的参数的值不可为空

参数值默认使用URL编码

示例:

//标准使用方式,默认使用URL编码@GET("/image/{id}")Call example(@Path("id")intid);//默认使用URL编码@GET("/user/{name}")Call encoded(@Path("name") String name);//不使用URL编码@GET("/user/{name}")Call notEncoded(@Path(value="name", encoded=true) String name);

1

2

3

4

5

6

7

8

9

1

2

3

4

5

6

7

8

9

2.8 POST注解:

用于发送一个POST请求

POST注解一般必须添加相对路径或绝对路径或者全路径,如果不想在POST注解后添加请求路径,则可以在方法的第一个参数中用@Url注解添加请求路径

2.9 PUT注解:

用于发送一个PUT请求

PUT注解一般必须添加相对路径或绝对路径或者全路径,如果不想在PUT注解后添加请求路径,则可以在方法的第一个参数中用@Url注解添加请求路径

3.0 Query注解:

作用于方法的参数

用于添加查询参数,即请求参数

参数值通过String.valueOf()转换为String并进行URL编码

使用该注解定义的参数,参数值可以为空,为空时,忽略该值,当传入一个List或array时,为每个非空item拼接请求键值对,所有的键是统一的,如: name=张三&name=李四&name=王五.

示例:

@GET("/list")Call list(@Query("page")intpage);@GET("/list")Call list(@Query("category") String category);//传入一个数组@GET("/list")Call list(@Query("category") String... categories);//不进行URL编码@GET("/search")Call list(@Query(value="foo", encoded=true) String foo);

1

2

3

4

5

6

7

8

9

10

1

2

3

4

5

6

7

8

9

10

3.1 QueryMap注解:

作用于方法的参数

以map的形式添加查询参数,即请求参数

参数的键和值都通过String.valueOf()转换为String格式

map的键和值默认进行URL编码

map中每一项的键和值都不能为空,否则抛出IllegalArgumentException异常

示例:

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

1

2

3

4

5

6

1

2

3

4

5

6

3.2 Streaming注解:

作用于方法

处理返回Response的方法的响应体,即没有将body()转换为byte []。

3.3 Url注解:

作用于方法参数

用于添加请求的接口地址

示例:

@GETCall list(@UrlString url);

1

2

1

2

注意事项:

1,以上部分注解真正的实现在ParameterHandler类中,,每个注解的真正实现都是ParameterHandler类中的一个final类型的内部类,每个内部类都对各个注解的使用要求做了限制,比如参数是否可空,键和值是否可空等.

2,FormUrlEncoded注解和Multipart注解不能同时使用,否则会抛出methodError(“Only one encoding annotation is allowed.”);可在ServiceMethod类中parseMethodAnnotation()方法中找到不能同时使用的具体原因.

3,Path注解与Url注解不能同时使用,否则会抛出parameterError(p, “@Path parameters may not be used with @Url.”),可在ServiceMethod类中parseParameterAnnotation()方法中找到不能同时使用的具体代码.其实原因也很好理解: Path注解用于替换url路径中的参数,这就要求在使用path注解时,必须已经存在请求路径,不然没法替换路径中指定的参数啊,而Url注解是在参数中指定的请求路径的,这个时候指定请求路径已经晚了,path注解找不到请求路径,更别提更换请求路径中的参数了.

4,对于FiledMap,HeaderMap,PartMap,QueryMap这四种作用于方法的注解,其参数类型必须为Map的实例,且key的类型必须为String类型,否则抛出异常(以PartMap注解为例):parameterError(p, “@PartMap keys must be of type String: ” + keyType);

5,使用Body注解的参数不能使用form 或multi-part编码,即如果为方法使用了FormUrlEncoded或Multipart注解,则方法的参数中不能使用Body注解,否则抛出异常parameterError(p, “@Body parameters cannot be used with form or multi-part encoding.”);

小结:

早就想把Retrofit的各个注解的含义和使用条件研究一番,但是几个月来一直加班,根本没多少时间仔细研读,中间有点闲暇时间也想偷个懒休息一下,眼看着已经是2016年最后一天了,再不研究就要等明年了,所以下定决心,花了大半天的时间,把Retrofit 2.1版本的源码彻底的读了一篇,边读变做笔记,读完之后把这些笔记又做了一些整理,才有了这2016年最后一篇博客.当然,文章中难免有错误或者不足之处,如果发现,还请及时告知,谢谢啦~~

读完源码,发现Retrofit并不像理想中的那么好,这里说的不好不是说代码架构不好,而是单指易用性这个方面,所有会用Retrofit的朋友,都知道如何使用Retrofit发送一个请求:

需要写至少一个接口

然后定义至少一个接口方法

然后构建Retrofit

然后调用create方法生成接口类

然后调用enqueue或者 execute方法发送一个异步或同步请求

一个简单的请求一共经历了5步,这还不算完:

- 你需要添加json解析器,如GsonConvertFactory,来自动序列化json串

- 你需要配置统一的cookie拦截器,这些代码需要你自己编写(网上复制粘贴的除外)

- 一般你还需要添加日志拦截器,因为在debug的时候你会发现,Retrofit竟然他妈的不能拼接出完整的url请求地址(完整的请求地址包括请求的主机地址,接口名,所有请求参数拼接,Retrofit最多只能看到接口,后面拼接的参数是看不到的,这在调试的时候很让人不爽)

如果你说这些都不是事,那么我们再看:

Retrofit提供了MultiPart注解,说明我们可以上传文件,又提供了Streaming注解,说明我们可以下载文件,我们知道Retrofit可以干这些事,但是我们还是没有办法直接写上传下载代码,这些东西都需要我们自己去封装,这也是为什么目前有很多基于Retrofit构建的二次封装库的原因

很多人用Retrofit基本上也就会发送一个get或者post请求,也就熟悉个别几个作用于参数的注解,如果你让这些人用Retrofit去搞定所有RESTful风格的接口,可能大部分人直接懵逼,因为他们不知道各个方法的注解和参数的注解怎么搭配使用才能完美运行,而不是抛出异常,因为Retrofit制定的这些规则你必须遵守

有些狂热的Retrofit粉丝大呼Retrofit有着插拔式设计,想用就用,想不用就不用,耦合很低,这确实是Retrofit的优点,但也正是因为足够灵活,导致易用性不够,不然也不会产生这么多基于Retrofit构建的框架了

算了,不喷了.毕竟一会就到2017年了,给2016年留下个好印象把,作为程序员,对新技术新东西保持强烈的好奇心的同时还应该保持一个清醒的头脑,不要盲目跟风.我去,说好不喷了,怎么又喷上了.

哎,看掘金上其他开发者发表的博客,都是对2016年一年的总结及2017年的展望,我2016年的最后一篇博客竟然是一篇纯纯的技术文章,无所谓啦,每个人的境遇不一样,谁让我的新年签是:简单,比2016过得好就行!

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

推荐阅读更多精彩内容

  • 本篇文章基于retrofit-2.1进行分析. 方法注解:@GET、@POST、@PUT、@DELETE、@PAT...
    丨Fan阅读 7,637评论 0 11
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,626评论 18 139
  • 整体Retrofit内容如下: 1、Retrofit解析1之前哨站——理解RESTful2、Retrofit解析2...
    隔壁老李头阅读 15,058评论 4 39
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,727评论 25 707
  • 她说。 阿姨,你真漂亮。 她说。 阿姨,你的眼睛是蓝色的,真漂亮。 …… 拥挤的公交车上,人们蹙着眉摩肩接踵。车门...
    蓝莫熙阅读 263评论 0 1