Retrofit初探

前言

Retrofit现在真的是炒鸡火,前不久用了之后,并且与rxjava配合得简直天衣无缝,我便深深的爱上了这个框架。

概述

Retrofit是Square公司开发,针对Android网络请求的框架。通过文档http://square.github.io/retrofit/可以看到它的介绍

A type-safe HTTP client for Android and Java

简单的说它就是一个HTTP请求工具,这里提到了类型安全,而Retrofit就是通过声明泛型来实现类型安全的,也就是说通过声明泛型能够约束最终我们想要的对象是什么。

用法

Retrofit 可以利用接口,方法和注解参数(parameter annotations)来声明式定义一个请求应该如何被创建。这里酷炫的注解参数也是我深深喜爱这个框架的原因之一。比如我们需要请求以下的API

https://api.github.com/repos/{owner}/{repo}/contributors

查看GitHub上"owner"的"repo"中的contributors,首先要创建一个以下的接口:

public interface GitHubService {
    @GET("/repos/{owner}/{repo}/contributors")
    Call<List<Contributor>> repoContributors(
        @Path("owner") String owner,
        @Path("repo") String repo);
}

然后通过Retrofit生成一个此接口的实现类:

Retrofit retrofit = new Retrofit.Builder()
    .baseUrl("https://api.github.com/")
    .build();

GitHubService service = retrofit.create(GitHubService.class);

默认的,Retrofit只能接收OkHttp'sRequestBody并转换成OkHttp'sResponseBody,不过在这里在创建Retrofit对象时,可以根据需要加入Converters,同时还需要在build.gradle中添加依赖。

  • Gson: com.squareup.retrofit2:converter-gson
  • Jackson: com.squareup.retrofit2:converter-jackson
  • Moshi: com.squareup.retrofit2:converter-moshi
  • Protobuf: com.squareup.retrofit2:converter-protobuf
  • Wire: com.squareup.retrofit2:converter-wire
  • Simple XML: com.squareup.retrofit2:converter-simplexml
  • Scalars (primitives, boxed, and String): com.squareup.retrofit2:converter-scalars

这里获取的是json数据,那么修改代码成:

Retrofit retrofit = new Retrofit.Builder()
    .baseUrl("https://api.github.com/")
    .addConverterFactory(GsonConverterFactory.create())
    .build();

接着获取接口中Call的实例:

Call<List<Contributor>> call = service.repoContributors("square", "retrofit");

最后就可以通过执行call获取到数据了:

call.enqueue(new Callback<List<Contributor>>() {
    @Override
    public void onResponse(Response<List<Contributor>> response) {
        for (Contributor contributor : response.body()) {
            System.out.println(contributor.login + ":" + contributor.contributions );
        }
    }
    @Override
    public void onFailure(Throwable t) {
    }
});

这里的Contributor就是我们创建的数据bean类,在此例子的api中,我们可以获取到很多数据,但是我们只需要筛选自己需要的数据创建一个bean类传入就可以了。

public class Contributor{
    String login;
    int contributions;
}

支持rxjava

当然,Retrofit还支持rxjava,这时我们需要在创建Retrofit实例的时候再加入一个adapter:

Retrofit retrofit = new Retrofit.Builder()
    .baseUrl("https://api.github.com/")
    .addConverterFactory(GsonConverterFactory.create())
    .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
    .build();

之前的接口也需要修改,这时候我们需要返回的不再是Call,而是Observable

public interface GitHubService {
    @GET("/repos/{owner}/{repo}/contributors")
    Observable<List<Contributor>> repoContributors(
        @Path("owner") String owner,
        @Path("repo") String repo);
}

获取到Observable实例之后就是一连串的rxjava的代码,比如子线程生产数据,主线程消费数据,更新UI等等。

service.repoContributors("square", "retrofit")
        .subscribeOn(Schedulers.io())
        .observeOn(AndroidSchedulers.mainThread())
        .subscribe(new Subscriber<List<Contributor>>() {
            @Override
            public void onStart() {
                showProgressBar();
            }

            @Override
            public void onCompleted() {
                dismissProgressBar();
            }

            @Override
            public void onError(Throwable e) {
                Log.i(">>>", "onError " + e.toString());
            }

            @Override
            public void onNext(List<Contributor> list) {
                for (Contributor contributor : list) {
                        System.out.println(contributor.login + ":" + contributor.contributions );
                }
            }
        });

注解词

前面说到了Retrofit可以通过注解参数的方式来声明定义一个请求,一下就来介绍其中重要的注解词用法。

GET请求

  1. @Query

    Get方法请求参数都会以key=value的方式拼接在url后面。

     Retrofit retrofit = new Retrofit.Builder()
             .baseUrl("https://api.douban.com/v2/")
             .addConverterFactory(GsonConverterFactory.create())
             .build();
    
     Call<BookBean> call = retrofit.create(DemoService.class).queryDemo(1);
     String url = call.request().url().toString();
     Log.i(">>>", "onCreate :" + url);
    

打印结果:</br>
https://api.douban.com/list?page=1

  1. @QueryMap

    如果拼接在后面的参数过多,可以通过QueryMap方式将所有的参数集成在一个Map集合中传递。

     Retrofit retrofit = new Retrofit.Builder()
             .baseUrl("https://api.douban.com/v2/")
             .addConverterFactory(GsonConverterFactory.create())
             .build();
    
     Map<String, String> options = new HashMap<>();
     options.put("start", "0");
     options.put("count", "3");
    
     Call<BookBean> call = retrofit.create(DemoService.class).queryMapDemo(options);
     String url = call.request().url().toString();
     Log.i(">>>", "onCreate :" + url);
    

打印结果:</br>
https://api.douban.com/v2/book/search?count=3&start=0

  1. @Path

    在url中的参数需要请求方提供的时候,可以利用path。

     Retrofit retrofit = new Retrofit.Builder()
             .baseUrl("https://api.douban.com/v2/")
             .addConverterFactory(GsonConverterFactory.create())
             .build();
    
     Call<BookBean> call = retrofit.create(DemoService.class).pathDemo(1);
     String url = call.request().url().toString();
     Log.i(">>>", "onCreate :" + url);
    

打印结果:</br>
https://api.douban.com/image/1

POST请求

  1. @Field

    POST请求需要把参数放置在请求体中,而不是拼接在url后面。

     Retrofit retrofit = new Retrofit.Builder()
             .baseUrl("https://api.douban.com/v2/")
             .addConverterFactory(GsonConverterFactory.create())
             .build();
    
     Call<BookBean> call = retrofit.create(DemoService.class).fieldDemo("zhangsan", "businessman");
    
    • 这里的注解还需要一个@FormUrlEncoded,将会自动将请求参数的类型调整为application/x-www-form-urlencoded。
  2. @FieldMap

    如果有更多的请求参数的时候,通过一个一个的参数传递就显得很麻烦而且容易出错,那么这个时候就可以用FieldMap

     Retrofit retrofit = new Retrofit.Builder()
             .baseUrl("https://api.douban.com/v2/")
             .addConverterFactory(GsonConverterFactory.create())
             .build();
    
     Map<String, String> fields = new HashMap<>();
     fields.put("start", "0");
     fields.put("count", "3");
    
     Call<BookBean> call = retrofit.create(DemoService.class).fieldMapDemo(fields);
    

Demo GitHub:https://github.com/cgzysan/RetrofitDemo</br>
同时我也用Retrofit写了一个Gnak.iohttps://github.com/cgzysan/Gank.io</br>
Retrofit未完待续!

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

推荐阅读更多精彩内容