Retrofit配合Rxjava实现网络请求和数据处理

前言

  • 本篇是Retrofit和Rxjava混合的的一个案例,配合Rxjava的操作符以及其API,讲解Retrofit怎么转化成Rxjava的使用类型实现简洁异步的网络请求
    代码托管地址

1. 依赖

    //retrofit2.0依赖 版本号一致
    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'


    //Rxjava依赖,可以将Retrofit的ServiceApi接口不仅返回call<T>还可以返回Observable<T>
    compile 'io.reactivex:rxjava:1.1.0'
    compile 'io.reactivex:rxandroid:1.1.0'

    compile 'net.orfjackal.retrolambda:retrolambda:1.8.1'
    compile 'de.greenrobot:eventbus:2.4.0'

    //在retrofit2.0中是没有日志功能的。但是retrofit2.0中依赖OkHttp,
    // 所以也就能够通过OkHttp中的interceptor来实现实际的底层的请求和响应日志。
    compile 'com.squareup.okhttp3:okhttp:3.4.1'
    compile 'com.squareup.okhttp3:logging-interceptor:3.4.1'

初始化Retrofit(着重addCallAdapterFactory)

  HttpLoggingInterceptor httpLoggingInterceptor = new HttpLoggingInterceptor(new HttpLoggingInterceptor.Logger() {
            @Override
            public void log(String message) {
                Log.i("Rxjava", message);
            }
        });
        httpLoggingInterceptor.setLevel(HttpLoggingInterceptor.Level.BODY);
        OkHttpClient okHttpClient = new OkHttpClient.Builder()
                .retryOnConnectionFailure(true)//设置重连
                .connectTimeout(15, TimeUnit.SECONDS)
                .addNetworkInterceptor(httpLoggingInterceptor)//网络拦截器(打印信息更丰富,消息实体内容长度类型等)
                .build();

备注:
* HttpLoggingInterceptor可以设置无参,这里Logger()设置定向过滤Rxjava,默认是okhttp
* 设置拦截范围:BODY--- 请求/响应行 + 头 + 体
* 拦截器分为两种: .addInterceptor(httpLoggingInterceptor)//应用拦截器
*/

   Retrofit retrofit = new Retrofit.Builder()
                .baseUrl(ENDPOINT)
                .client(okHttpClient)
                .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
                .addConverterFactory(GsonConverterFactory.create())
                .build();
        mGithubAPI = retrofit.create(GithubAPIInterface.class);

备注:
* 1.基地址必须有(可以是全部,也可以是部分)
* 2.添加拦截器(非必须)
* 3.Converter是对于Call<T>中T的转换, Call<ResponseBody>--------->Call<Poju>
* Call<T>中的Call也是可以被替换的,而返回值的类型就决定你后续的处理程序逻辑(非必须)
* 4.CallAdapter可以对Call转换,Retrofit提供了多个CallAdapter,这里以RxJava的为例,用Observable<T>代替Call<T>(非必须)
* 即Retrofit创建出接口的代理对象有多种形式,Call<T>、Observable<T>而后者就是Retrofit实现对Rxjava的支持
* 5.初始化Retrofit
* 6.用Retrofit创建出接口的代理对象,用代理对象来操作其方法,返回Call<Poju>
* 通过Call<Poju>来请求入队,execute/enqueue(同步/异步)
*
*/

Retrofit配合Rxjava实现网络请求

备注:这里先上代码,后面会一一解释
demo主要基于一个网络返回的信息做下一次网络请求,对比上一篇纯Retrofit的请求处理
完整address1:https://api.github.com/users?per_page=1
完整address2:https://api.github.com/users/mojombo

 final Observer subscriber = new Observer<List<GithubUserDetail>>() {
            @Override
            public void onCompleted() {
                Log.i("tag", "onCompleted-----rx----");
            }

            @Override
            public void onError(Throwable e) {
                Log.i("tag", "onError---rx------");
            }

            @Override
            public void onNext(List<GithubUserDetail> githubUserDetails) {
                Log.i("tag", "onNext-----rx----");
                tv.setText(githubUserDetails.toString());
            }
        };
        subscribe = new GithubServiceManager().rxFetchUserDetails().subscribe(subscriber);
  • 单例创建接口的代理对象
   private GithubService() {
        HttpLoggingInterceptor httpLoggingInterceptor = new HttpLoggingInterceptor(new HttpLoggingInterceptor.Logger() {
            @Override
            public void log(String message) {
                Log.i("Rxjava", message);
            }
        });
        httpLoggingInterceptor.setLevel(HttpLoggingInterceptor.Level.BODY);
        OkHttpClient okHttpClient = new OkHttpClient.Builder()
                .retryOnConnectionFailure(true)//设置重连
                .connectTimeout(15, TimeUnit.SECONDS)
                .addNetworkInterceptor(httpLoggingInterceptor)//网络拦截器(打印信息更丰富,消息实体内容长度类型等)
                .build();

        Retrofit retrofit = new Retrofit.Builder()
                .baseUrl(ENDPOINT)
                .client(okHttpClient)
                .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
                .addConverterFactory(GsonConverterFactory.create())
                .build();
        mGithubAPI = retrofit.create(GithubAPIInterface.class);
    }


    /**
     * 单例
     */
    private static final GithubService INSTANCE = new GithubService();

    public static GithubService getInstance() {
        return INSTANCE;
    }
  • 接口内嵌方法
    public Observable<GithubUserDetail> rxRequestUserDetails(@Path("username") String username) {
        Log.i("tag", "rxRequestUserDetails----------");
        return mGithubAPI.rxRequestUserDetails(username);
    }

    public Observable<List<GithubUser>> rxRequestUsers() {
        return mGithubAPI.rxRequestUsers(PER_PAGE);//第一条
    }
    
    @GET("users")//@Query是以Key = value的形式拼接到基地址
    Observable<List<GithubUser>> rxRequestUsers(@Query("per_page") Integer perPage);
    
    @GET("users/{username}")//@Path以参数的形式传递
    Observable<GithubUserDetail> rxRequestUserDetails(@Path("username") String username);
  • Observable的关键实现
    public Observable<List<GithubUserDetail>> rxFetchUserDetails() {
        return mService.rxRequestUsers()
                .concatMap(new Func1<List<GithubUser>, Observable<? extends GithubUser>>() {
                    @Override
                    public Observable<? extends GithubUser> call(List<GithubUser> iterable) {
                        Log.i("tag", "---------------concatMap1");//根据接收情况,这里只会执行一次
                        return Observable.from(iterable);
                    }
                })//这里Observable激活后,根据接收情况会一一按顺序发送事件,如果这里new Func1参数GithubUser变成List<GithubUser>就执行一次
                .concatMap(new Func1<GithubUser, Observable<? extends GithubUserDetail>>() {
                               @Override
                               public Observable<? extends GithubUserDetail> call(GithubUser githubUser) {
                                   Log.i("tag", "---------------concatMap2---+"+githubUser.mLogin);
                                   //继续根据githubUser返回再次请求,获取具体信息
                                   return mService.rxRequestUserDetails(githubUser.mLogin);
                               }
                           }
                )
                .toList()//上一步返回的一个个的GithubUser的信息,通过toList()将其转化成一个事件队列的Observable,不经过这一步则订阅的时候
                //只会接收List<GithubUser>的最后一个对象
                .subscribeOn(Schedulers.newThread())
                .observeOn(AndroidSchedulers.mainThread());
    }

讲解:mService.rxRequestUsers()返回 以List形式的Observable对象,
new Func1(参数,返回对象Observable<? extends GithubUser>可以写成Observable< GithubUser>),
concatmap是配合Func1()来实现整个事件队列对象的转换,根据new Func1()的参数List<GithubUser>里的事件序列创建出一个的Observable<GithubUser>对象,
这里的对象和返回的observable不是一个对象,更加的解耦了上下关联,通过from将事件序列汇总到Observable,单一线程创建所以,顺序不变

  • 订阅,处理数据
  final Observer subscriber = new Observer<List<GithubUserDetail>>() {
            @Override
            public void onCompleted() {
                Log.i("tag", "onCompleted-----rx----");
            }

            @Override
            public void onError(Throwable e) {
                Log.i("tag", "onError---rx------");
            }

            @Override
            public void onNext(List<GithubUserDetail> githubUserDetails) {
                Log.i("tag", "onNext-----rx----");
                tv.setText(githubUserDetails.toString());
            }
        };
        subscribe = new GithubServiceManager().rxFetchUserDetails().subscribe(subscriber);
        //执行List.size()次onNext()(不出错的情况下 ),最终执行onCompleted()
        //这里实例化Subscription subscribe;是为了使用完毕是及时反订阅
  • 反订阅,防止内存泄漏
  @Override
    protected void onDestroy() {
        if (subscribe != null && subscribe.isUnsubscribed()) {
            subscribe.unsubscribe();
        }
    }

总结

通过Retrofit和Rxjava的混合使用和上上篇纯Retrofit使用优点显而易见,解耦了各个模块,避免因上级返回信息再次做处理时的层层嵌套,代码简洁干练,大大提高了读写行!
代码托管地址

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

推荐阅读更多精彩内容