Android 快速开发系列之网络篇(Retrofit 2.0)

网络请求库有很多优秀的开源项目okhttp,volley都是很不错的,但是个人比较喜欢Retrofit。原因有以下几点:
1.一个类型安全的REST客户端 。
2.通过GsonConverter可以直接把服务器响应的json字符串映射成对象,这一切都是自动化的。当然还有其他的转换器并且支持自定义非常的灵活。
3.支持同步请求和异步请求
4.2.0开始加入对Rxjava的支持,配合Rxjava编程爽爆了,代码变的很清晰。
5.2.0开始可以很轻松的取消请求,你只需调用call.cancel()
6.性能和速度都比volley等更好。

重要的事情强调一下本文针对的是Retrofit 2.0讲的,Retrofit 1.xx的版本和2.0版本有许多改动。

retrofit2.0 知识点

步骤:
1.首先导入需要的库
compile 'com.squareup.retrofit2:retrofit:2.0.2'
compile 'com.squareup.retrofit2:converter-gson:2.0.2'

2.新建一个interface接口,用于存放和服务器交互的接口,针对常用操作下面各写一个范例

  • Get无参数请求
      public interface MeizhiApi {
      @GET("data/福利/10/{day}")//()里面的是相对路径,当然绝对路径也是可以的
      public Call<BeanMezhi> getMeizhi(@Path("day") int day);//{}里面的是要替换的内容 用注解@Path映射
    

}

- Get单个参数请求
```java
  public interface SmsApi {
  @GET("http://xxx/getSmsCode")
  public Call<JSONObject> getSms(@Query("tel") String tel);//相当于http://xxx/getSmsCode?tel="xx"
}
  • Get多个参数请求
    public interface SmsApi {
      @GET("http://weixing.wxspider.com:8087/appVoip!getSmsCode")
      public Call<JSONObject> getSms(@QueryMap Map<String, String> options);
    

}

- Post请求
```java
public interface DemoApi {
  @POST("checkupdate")
  public Call<BeanVersion> getVersion(@Body HashMap type);//注意这里用的是@Body 
}
  • 表单请求

public interface DemoApi {
@FormUrlEncoded
@POST("user/edit")
Call<User> updateUser(@Field("first_name") String first, @Field("last_name") String last);
}

- 文件上传
```java
public interface DemoApi {
  @Multipart
  @PUT("user/photo")
  Call<User> updateUser(@Part("photo") RequestBody photo, @Part("description") RequestBody description);
}

注意:上面所有的接口返回的都是Call类型的这是和1.x版本不同的

3.实例化上面的接口

 public class ApiServiceManager {
 private static ApiServiceManager ourInstance = new ApiServiceManager();

 public static ApiServiceManager getInstance() {
     return ourInstance;
 }

 private DemoApi demoApi;
 private MeizhiApi meizhiApi;
 private SmsApi smsApi;

 private ApiServiceManager() {
     Retrofit.Builder builderWithGson = new Retrofit.Builder()
             .baseUrl(Config.DemoBaseUrl)//设置基础url
             .addConverterFactory(GsonConverterFactory.create());//Gson转换器直接返回对象
     Retrofit retrofit = builderWithGson.build();
     demoApi = retrofit.create(DemoApi.class);//拿到和服务器交互的接口实例
     builderWithGson.baseUrl(Config.MeizhiBaseUrl);
     retrofit = builderWithGson.build();
     meizhiApi = retrofit.create(MeizhiApi.class);
     Retrofit.Builder builderWithJson = new Retrofit.Builder()
             .addConverterFactory(JsonConverterFactory.create());//Json转换器返回JSONObject,因为有些接口返回的数据很简单不想写个Bean
     retrofit = builderWithJson.build();
     smsApi = retrofit.create(SmsApi.class);
 }

 public DemoApi getDemoApi() {
     return demoApi;
 }

 public MeizhiApi getMeizhiApi() {
     return meizhiApi;
 }

 public SmsApi getSmsApi() {
     return smsApi;
 }
}

4.使用接口

  • 同步请求使用方式
     new Thread(new Runnable() {
              @Override
              public void run() {
                  Call<BeanMezhi> mezhiCall = ApiServiceManager.getInstance().getMeizhiApi().getMeizhi(1);
                  Response<BeanMezhi> response = null;//同步请求会阻塞线程,因此你不能在安卓的主线程中调用,不然会面临NetworkOnMainThreadException,想调用execute方法,请在后台线程执行
                  try {
                      response = mezhiCall.execute();
                      if (response.body() != null) {//如果不能解析成对应的实体BeanMezhi则response.body()的值是空
                          Log.d(tag, new Gson().toJson(response.body()).toString());
                      }
                  } catch (IOException e) {
                      e.printStackTrace();
                  }
              }
          }).start();
    
  • 异步请求使用方式
    HashMap map = new HashMap();
        map.put("ostype", "android");
        Call<BeanVersion> versionCall = ApiServiceManager.getInstance().getDemoApi().getVersion(map);
        versionCall.enqueue(new Callback<BeanVersion>() {//异步请求
            @Override
            public void onResponse(Call<BeanVersion> call, Response<BeanVersion> response) {//回调运行在主线程
                if (response.body() != null) {
                    Log.d(tag, new Gson().toJson(response.body()).toString());
                }
            }
    
            @Override
            public void onFailure(Call<BeanVersion> call, Throwable t) {
                Log.d(tag, "onFail");
            }
        });
    

5.取消正在进行中的业务

 call.cancel();
  ```

基本使用介绍完了,如果需要自定义Converter或者自定义CallAdapter,那么请继续往下看。

---
**1. Converter** 官方提供的转换器有

Gson: com.squareup.retrofit:converter-gson

Jackson: com.squareup.retrofit:converter-jackson

Moshi: com.squareup.retrofit:converter-moshi

Protobuf: com.squareup.retrofit:converter-protobuf

Wire: com.squareup.retrofit:converter-wire

Simple XML: com.squareup.retrofit:converter-simplexml

如何自定义转换器?
你也可以通过实现Converter.Factory接口来创建一个自定义的converter。以JsonConverter示例:
```java
public class JsonConverterFactory extends Converter.Factory {

  public static JsonConverterFactory create() {
      return new JsonConverterFactory();
  }

  @Override
  public Converter<ResponseBody, ?> responseBodyConverter(Type type, Annotation[] annotations, Retrofit retrofit) {
      return new JsonResponseBodyConverter<JSONObject>();
  }

  @Override
  public Converter<?, RequestBody> requestBodyConverter(Type type, Annotation[] parameterAnnotations, Annotation[] methodAnnotations, Retrofit retrofit) {
      return new JsonRequestBodyConverter<JSONObject>();
  }
}
final class JsonRequestBodyConverter<T> implements Converter<T, RequestBody> {
  private static final MediaType MEDIA_TYPE = MediaType.parse("application/json; charset=UTF-8");

  JsonRequestBodyConverter() {

  }

  public RequestBody convert(T value) throws IOException {
      return RequestBody.create(MEDIA_TYPE, value.toString());
  }
}
 final class JsonResponseBodyConverter<T> implements Converter<ResponseBody, T> {

   JsonResponseBodyConverter() {

   }

   @Override
   public T convert(ResponseBody value) throws IOException {
       JSONObject jsonObj;
       try {
           jsonObj = new JSONObject(value.string());
           return (T) jsonObj;
       } catch(JSONException e) {
           return null;
       }
   }
}

2. CallAdapter

在interface接口定义中,retrofit2.0默认都是返回的Call<T>模式的,如果我们想返回其他的类型也是可以,retrofit已经为Rxjava粉丝们准备了CallAdapter,它将作为Observable返回。使用它必须引入以下两个库

compile 'io.reactivex:rxandroid:1.2.0'
compile 'com.squareup.retrofit2:adapter-rxjava:2.0.2'

并在Retrofit Builder链表中调用addCallAdapterFactory

Retrofit.Builder builderWithGson = new Retrofit.Builder()
              .baseUrl(Config.DemoBaseUrl)//设置基础url
              .addConverterFactory(GsonConverterFactory.create())//Gson转换器直接返回对象
              .addCallAdapterFactory(RxJavaCallAdapterFactory.create());//增加RxjavaCallAdapter
Retrofit retrofit = builderWithGson.build();

接下来看怎么结合Rxjava调用接口

  //定义接口
  @GET("data/福利/10/{day}")
  public Observable<BeanMezhi> getMeizhi2(@Path("day") int day);

  //调用接口
   Observable<BeanMezhi> observable = ApiServiceManager.getInstance().getMeizhiApi().getMeizhi2(1);
   observable.subscribeOn(Schedulers.io())//获取数据指定运行在io线程
             .observeOn(AndroidSchedulers.mainThread())//发布到android主线程
             .subscribe(new Action1<BeanMezhi>() {
                  @Override
                  public void call(BeanMezhi beanMezhi) {
                      Log.d(tag, new Gson().toJson(beanMezhi).toString());//处理数据这里已经是运行在主线程了
                  }
              });

在使用rxjava以后我们不在需要写new Thread().start 这些“脏”代码了,rxjava对线程的调度非常强大,那么有同学会说感觉还不如返回Call<T>模式的然后异步调用来的简单。好吧,如果要对返回的数据先过滤,在排序,还要存储数据库等等一系列处理,你该怎么办?有了rxjava处理这种复杂的数据流一切就会变的简单清晰,如果在配合lambda表达式可以说是如虎添翼。

项目主页: http://square.github.io/retrofit/
JSONCoverter:https://github.com/brokge/Retrofit2.0-JSONCoverter

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

推荐阅读更多精彩内容