Retrofit与RxJava结合并进行封装

作者: 龙腾九天ly | 来源:发表于2017-01-04 13:44 被阅读150次

    博主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);  
     }  
    } 
    

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

    相关文章

      网友评论

        本文标题:Retrofit与RxJava结合并进行封装

        本文链接:https://www.haomeiwen.com/subject/uimovttx.html