Retrofit与RxJava结合并进行封装

博主CSDN昵称:守护者ly,欢迎大家前去指点

Retrofit和RxJava的优点有哪些在此就不赘述了,之前已经有数位大神在各个网站上发了若干文章,博主深感受益匪浅。 我在这里仅仅介绍一下我最近尝试的封装方法,至于为啥要封装,答:因为懒。。。
我们可以看一下使用 retrofit进行Http请求的代码:

    okHttpClient = new OkHttpClient.Builder()  
            .addInterceptor(httpLoggingInterceptor)  
            .build();  
    retrofit = new Retrofit.Builder()  
            .baseUrl(baseUrl)  
            .addCallAdapterFactory(RxJavaCallAdapterFactory.create())  
            .addConverterFactory(GsonConverterFactory.create())  
            .client(okHttpClient)  
            .build();  
    retrofitMethod = retrofit.create(RetrofitMethod.class);  

    Call<List<User>> call = retrofitMethod.processList();  
    call.enqueue(new Callback<List<User>>() {  
    @Override  
    public void onResponse(Call<List<User>> call, Response<List<User>> response) {  
      if(response.body().size() > 0){  
        List<User> lt_userList = response.body();  
        Gson gson = new Gson();  
        String st_userList = gson.toJson(lt_userList);  
        Log.d("MainActivity", "result = " + st_userList);  
        tv_showText.setText(st_userList);  
      }else{  
        Log.d("MainActivity", "no data");  
      }  
    }  

    @Override  
    public void onFailure(Call<List<User>> call, Throwable t) {  
      Log.d("MainActivity", "failed");  
    }  
  });  

当然,怎么可以少了build.gradle:

compile 'com.squareup.retrofit2:retrofit:2.1.0'  
compile 'com.squareup.retrofit2:converter-gson:2.1.0'  
compile 'com.squareup.retrofit2:adapter-rxjava:2.1.0'  
compile 'com.squareup.okhttp3:okhttp:3.4.1'  
compile 'com.squareup.okhttp3:logging-interceptor:3.4.1'  
compile 'com.squareup.okhttp3:okhttp-urlconnection:3.4.1'  
compile 'io.reactivex:rxjava:1.1.0'  
compile 'io.reactivex:rxandroid:1.1.0'

好了,一次请求看着还不错,至少感觉比异步任务要好一些,不过如果我再请求一个Result这个model的List呢?那么第二段代码又要写一次,将里面所有的User换成Result即可,but,总有你恶心的那一天,而且如果结合RxJava后,你会发现那个长度更恶心。。。
RxJava中会有指定观察者、被观察者、订阅等,还有指定在某个线程进行操作,这些内容大同小异,可以封装起来,于是乎有了第一版封装:

public void processLists(Subscriber<List<User>> subscriber){  
    retrofitMethod.processLists()  
        .subscribeOn(Schedulers.io())  
        .unsubscribeOn(Schedulers.io())  
        .observeOn(AndroidSchedulers.mainThread())  
        .subscribe(subscriber);  
} 

RetrofitMethod.java

@GET("user/list/")  
Observable<List<User>> processLists();

调用过程:

private void getLists(){  
  Subscriber<List<User>> subscriber = new Subscriber<List<User>>() {  
    @Override  
    public void onCompleted() {  
        Toast.makeText(RetrofitThree.this, "complete", Toast.LENGTH_SHORT).show();  
    }  

    @Override  
    public void onError(Throwable e) {  
        Toast.makeText(RetrofitThree.this, "error", Toast.LENGTH_SHORT).show();  
        Log.d("RetrofitThree", "error:" + e);  
    }  

    @Override  
    public void onNext(List<User> users) {  
        Gson gson = new Gson();  
        String st_userList = gson.toJson(users);  
        tv_resultThree.setText(st_userList);  
    }  
  };  
  HttpManager3.getInstanc().processLists(subscriber);  
}

RxJava的订阅者中的方法主要有三个:onComplete(),当不再有新的onNext()发出时,触发onComplete()作为队列完结;onError(),事件队列异常,当它被触发时,队列自动终止,原则上讲onError()与onComplete()只能触发一个,并且是队列的最后一个;onNext(),回调的处理写在这里。
这次封装我们将关心的事件都放在明显的位置,准备做处理,将RxJava的一些设置封装了起来。但是细心的童鞋会发现:封装你妹啊!就封了一个方法!哥要调多个方法怎么办?!都封里面哥就疯了!!!
额,这个事情博主也考虑了,所以正确的姿势应该是这样的:

public void processList(Observable observable , Subscriber subscriber){  
  observable.subscribeOn(Schedulers.io())  
        .unsubscribeOn(Schedulers.io())  
        .observeOn(AndroidSchedulers.mainThread())  
        .subscribe(subscriber);  
} 

调用方法:

Subscriber<List<User>> subscriber = new Subscriber<List<User>>() {  
  @Override  
  public void onCompleted() {  
    Toast.makeText(RetrofitThree.this, "complete", Toast.LENGTH_SHORT).show();  
  }  

  @Override  
  public void onError(Throwable e) {  
    Log.d("RetrofitThree", "error" + e);  
  }  

  @Override  
  public void onNext(List<User> users) {  
    Toast.makeText(RetrofitThree.this, "ok", Toast.LENGTH_SHORT).show();  
    Gson gson = new Gson();  
    tv_resultThree.setText(gson.toJson(users));  
  }  
};  
HttpManager3.getInstanc().processList(retrofitMethod.processLists() , subscriber);  

诸位现在感觉怎么样?这次封装的套路没变,只不过是同时接收了被观察者和订阅者(观察者),这样就不会十个方法封十次了。博主赶脚这次不会被打死了。
细心的小盆友又会说啦,介个onComplete()和onError()每次处理的方法基本都一样啊,我就关心onNext()。好吧,应大家之邀,把onComplete()和onError()也封装起来吧:

public class NewSubscriber3<T> extends Subscriber<T> {  

  //    回调接口  
  private HttpOnNextListener3 mSubscriberOnNextListener;  
  //    弱引用防止内存泄露  
  private WeakReference<Context> mActivity;  

  private Context context;  

  public NewSubscriber3(HttpOnNextListener3 mSubscriberOnNextListener, Context context) {  
    this.mSubscriberOnNextListener = mSubscriberOnNextListener;  
    this.mActivity = new WeakReference<>(context);  
    this.context = context;  
  }  

  @Override  
  public void onCompleted() {  
    Toast.makeText(context, "subscriber completed", Toast.LENGTH_SHORT).show();  
  }  

  @Override  
  public void onError(Throwable e) {  
    Log.d("NewSubscriber3", "error:" + e);  
  }  

  @Override  
  public void onNext(T t) {  
    if (mSubscriberOnNextListener != null) {  
        mSubscriberOnNextListener.onNext(t);  
    }  
  }  
}  

interface:

public interface HttpOnNextListener3<T>{  
    void onNext(T t);  
}  

调用过程:

    HttpOnNextListener3<List<User>> simpleOnNextListener = new HttpOnNextListener3<List<User>>() {  
        @Override  
        public void onNext(List<User> users) {  
            Toast.makeText(RetrofitThree.this, "ok3", Toast.LENGTH_SHORT).show();  
            Gson gson = new Gson();  
            tv_resultThree.setText(gson.toJson(users));  
        }  
    };  

    RetrofitMethod retrofitMethod = HttpManager3.getMethod();  
    NewSubscriber3<List<User>> newSubscriber3 =  
            new NewSubscriber3(simpleOnNextListener , this);  
    HttpManager3.getInstanc().processList(retrofitMethod.processLists() , newSubscriber3);

最后再奉上神秘的HttpManager3.java:

package com.example.administrator.mytest.Retrofit3;  

import com.example.administrator.mytest.Config.JavaConfig;  
import com.example.administrator.mytest.HttpClient.RetrofitMethod;  
import com.example.administrator.mytest.model.User;  

import java.util.List;  
import rx.Observable;  
import java.util.concurrent.TimeUnit;  

import okhttp3.OkHttpClient;  
import retrofit2.Retrofit;  
import retrofit2.adapter.rxjava.RxJavaCallAdapterFactory;  
import retrofit2.converter.gson.GsonConverterFactory;  
import rx.Subscriber;  
import rx.android.schedulers.AndroidSchedulers;  
import rx.schedulers.Schedulers;  

/** 
 * Created by Ly on 2017/1/3. 
 */  

public class HttpManager3 {  

private volatile static HttpManager3 INSTANCE;  
private static long DEFUALT_TIMEOUT = 5000;  
private volatile static RetrofitMethod retrofitMethod;  

  //构造方法私有  
  private HttpManager3(){  

  }  

  private static RetrofitMethod Retro(){  
    //手动创建一个OkHttpClient并设置超时时间  
    OkHttpClient.Builder builder = new OkHttpClient.Builder();  
    builder.connectTimeout(DEFUALT_TIMEOUT, TimeUnit.SECONDS);  

    Retrofit retrofit = new Retrofit.Builder()  
            .client(builder.build())  
            .addCallAdapterFactory(RxJavaCallAdapterFactory.create())  
            .addConverterFactory(GsonConverterFactory.create())  
            .baseUrl(JavaConfig.kfb_baseUrl)  
            .build();  

    retrofitMethod = retrofit.create(RetrofitMethod.class);  
    return  retrofitMethod;  
  }  

  //获取单例  
  public static HttpManager3 getInstanc(){  
    if(INSTANCE == null){  
        synchronized (HttpManager3.class){  
            if(INSTANCE == null){  
                INSTANCE = new HttpManager3();  
            }  
        }  
    }  
    return INSTANCE;  
  }  

  public static RetrofitMethod getMethod(){  
    if(retrofitMethod == null){  
        synchronized (Retro()){  
            if(retrofitMethod == null){  
                retrofitMethod = Retro();  
            }  
        }  
    }  
    return retrofitMethod;  
  }  

  public void processLists(Subscriber<List<User>> subscriber){  
    retrofitMethod.processLists()  
            .subscribeOn(Schedulers.io())  
            .unsubscribeOn(Schedulers.io())  
            .observeOn(AndroidSchedulers.mainThread())  
            .subscribe(subscriber);  
  }  

  public void processList(Observable observable , Subscriber subscriber){  
    observable.subscribeOn(Schedulers.io())  
            .unsubscribeOn(Schedulers.io())  
            .observeOn(AndroidSchedulers.mainThread())  
            .subscribe(subscriber);  
 }  
} 

博主的现学现卖到此告一段落,如有不足之处,还望大家指出~

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

推荐阅读更多精彩内容