美文网首页Andorid的好东西android网络开发网络
Rxjava +Retrofit 你需要掌握的几个技巧,Retr

Rxjava +Retrofit 你需要掌握的几个技巧,Retr

作者: Tamic | 来源:发表于2016-08-05 20:21 被阅读18851次

本文出处 :Tamic
文/ http://www.jianshu.com/p/b1979c25634f

Rxjava +Rterofit 需要掌握的几个技巧

这里写图片描述

RxJava入门和详解请移步 比较有名的《RxJAVA详解》,这里继续前篇一系列的介绍一些容易忽略的技巧.

Retrofit+RxJava结合系列请阅读:


取消订阅

一般我们在视图消亡后,无需RxJava再执行,可以直接取消订阅

subscription.unsubscribe()

 observable.unsubscribeOn(Schedulers.io());

可用在activity的 onDestroy(), Fragment的 onDestroyView()中调用

还有种场景是借助rxJava请求网络数据,需要网络返回后保存数据并更新UI,这种情况视图已经消亡了必定会导致rxJava出错,导致App闪退,这种我们可以判断前的activity/view是否为空,并是否已showing,如果
两者都不存在,即可无须更新UI。只处理保存数据即可。

订阅问题

需要UI绘制后再进行订阅的场景,防止阻塞UI,我们需要延迟订阅执行。

立即订阅;

         observable
               .subscribeOn(Schedulers.io())
               .observeOn(AndroidSchedulers.mainThread())
               .subscribe(action);

延迟订阅

observable.delay(2, TimeUnit.SECONDS)
               .subscribeOn(Schedulers.io())
               .observeOn(AndroidSchedulers.mainThread())
               .subscribe(action);

基础ApiService

通常我们写接口会有以下定义,增加一个api就必须写一个方法

public interface MyApi {

@GET("app.php")
Observable<SouguBean> getSougu(@Query("name") String name);

@GET("/getWeather")
Observable<ResponseBody> getWeather(@QueryMap Map<String, String> maps);
}

很多时候每新增一个接口就要写一个api,是不是有很好的方法代替这种情况。

@GET()
<T> Observable<ResponseBody> get(
        @Url String url,
        @QueryMap Map<String, T> maps);

我们可以定义一个通用的getApi,将url动态传入,返回Modle定义为ResponseBody, 并将实际参数定义为泛型,不管是更改url,还是服务端返回类型,包括参数个数都可以完美适配,这种方式技术不到位的千万别用,因为Retrofit明确说明接口必须要给定明确类型,悠着点哈!

上层进行通用组装时就可以这样子:

   public <T> T get(String url, Map<String, T> maps, BaseSubscriber<ResponseBody> subscriber) {
return (T) apiManager.get(url, maps)
.compose(schedulersTransformer)
.compose(handleErrTransformer())
.subscribe(subscriber);
}

看不懂?看不懂不算奇怪,源码可以去文章末尾下载研究,这里只是列举了一下。这种方式很适合从HttpClent迁移到Retrofit带来接口适配问题,一用一个准啊…

基础Subscriber

很多时候我们需要借用RxJava开启多个observable去读取网络,这是我们对不同Subscriber处理起来比较麻烦,因此统一对Subscriber对网络返回进行处理和, 有无网络做判断,甚至可以根据需求显示加载进度等
构建抽象的BaseSubscribe类,只处理start()onCompleted() ,上层处理时只处理onError()onNext()

 /**
    * BaseSubscriber
    * Created by Tamic on 2016-7-15.
    */
   public abstract class BaseSubscriber<T> extends Subscriber<T> {

       private BaseActivity context;

       public BaseSubscriber(BaseActivity context) {
           this.context = context;
       }

       @Override
       public void onStart() {
           super.onStart();

           if (!NetworkUtil.isNetworkAvailable(context)) {

               Toast.makeText(context, "当前网络不可用,请检查网络情况", Toast.LENGTH_SHORT).show();
              // **一定要主动调用下面这一句**
               onCompleted();

               return;
           }
           // 显示进度条
           showLoadingProgress();
       }

       @Override
       public void onCompleted() {
          //关闭等待进度条
          closeLoadingProgress();

       }


    }

这样我们上层调用时只关心成功和失败即可,无需再关心网络情况

```
observable..subscribeOn(Schedulers.io())
               .observeOn(AndroidSchedulers.mainThread())
               .subscribe(new BaseSubscriber<ResponseBody>(MainActivity.this) {

                @Override
                public void onError(Throwable e) {
                   Toast.makeText(MainActivity.this, e.getMessage(), Toast.LENGTH_LONG).show();
                }

                @Override
                public void onNext(ResponseBody responseBody) {
                    Toast.makeText(MainActivity.this, responseBody.toString(), Toast.LENGTH_LONG).show();
                }
            });
  );

如果想对Error错误统一处理,也可以在BaseSubscriber处理onError(), 然后回调搭到callback上层,具体看自己项目情况而定,可以接着往下看

如果对成功结果进行处理,则可以将ResonseBody加入泛型<Response<T>> , Response一般是包含Code,MSg, Data的,在这里你可以根据判断code来进行业务分发,代码很简单,具体看文章结尾源码即可

如果你觉得目前的返回判断麻烦,也可以定义Response基类

 /**
* 网络返回基类 支持泛型
* Created by Tamic on 2016-06-06.
*/

public class BaseResponse<T> {

  private int code;
  private String msg;
  private T data;

 public int getCode() {
     return code;
 }

public void setCode(int code) {
    this.code = code;
}

public String getMsg() {
    return msg;
}

public void setMsg(String msg) {
    this.msg = msg;
}

public T getData() {
    return data;
}

public void setData(T data) {
    this.data = data;
}

public boolean isOk() {
    return code == 0;
}

}

这样我们在onNext() 只需统一判断状态码即可

                       @Override
                        public void onNext(BaseResponse<IpResult> responseBody) {

                            if (responseBody.isOk()) {
                           //这里这个ok不是http访问的ok,

//是和服务器约定好的成功码 有的人不喜欢可以不用加这个筛选, 也有的人喜欢将业务加到业务回调中,如果不是成功码 也//不走错误回调,也不走成功回调,直走业务回调

                                IpResult ip = responseBody.getData();
                                Toast.makeText(MainActivity.this, ip.toString(), Toast.LENGTH_LONG).show();
                            }

                        }

错误结果问题

通过RXJva的 Func1来进行对原始的Throwable 进行包装转换

我们将原来Throwable 强转成自定义的 ResponeThrowable;

private static class HttpResponseFunc《T》 implements Func1《Throwable, Observable《T》》 {
    @Override public Observable<T> call(Throwable t) {
        return Observable.error(ExceptionHandle.handleException(t));
    }
}

ResponeThrowable

public static class ResponeThrowable extends Exception {
    public int code;
    public String message;

    public ResponeThrowable(Throwable throwable, int code) {
        super(throwable);
        this.code = code;

    }
}

我们已经处理好强转工作后 继续把 Func1加到Observable 中:

因此这样用observable提供的onErrorResumeNext 则可以将你自定义的Func1 关联到错误处理类中:

  ((Observable) observable).onErrorResumeNext(new HttpResponseFunc<T>());

很可能你感觉有点不理解,这前提你需要了解RxJava的转义符和操 Observable.Transformer
还有Func1

这样我们对服务器返回的错误状态进行了自我的处理,再稍加翻译下便可以达到用户看懂的语言

这个类我参考一叶扁舟同学的案列,我再次做了改进:

ExceptionHandle 错误处理驱动

public class ExceptionHandle {

 private static final int UNAUTHORIZED = 401;
 private static final int FORBIDDEN = 403;
 private static final int NOT_FOUND = 404;
 private static final int REQUEST_TIMEOUT = 408;
 private static final int INTERNAL_SERVER_ERROR = 500;
 private static final int BAD_GATEWAY = 502;
 private static final int SERVICE_UNAVAILABLE = 503;
 private static final int GATEWAY_TIMEOUT = 504;

 public static ResponeThrowable handleException(Throwable e) {
    ResponeThrowable ex;
    if (e instanceof HttpException) {
        HttpException httpException = (HttpException) e;
        ex = new ResponeThrowable(e, ERROR.HTTP_ERROR);
        switch (httpException.code()) {
            case UNAUTHORIZED:
            case FORBIDDEN:
            case NOT_FOUND:
            case REQUEST_TIMEOUT:
            case GATEWAY_TIMEOUT:
            case INTERNAL_SERVER_ERROR:
            case BAD_GATEWAY:
            case SERVICE_UNAVAILABLE:
            default:
                ex.message = "网络错误";
                break;
        }
        return ex;
    } else if (e instanceof ServerException) {
        ServerException resultException = (ServerException) e;
        ex = new ResponeThrowable(resultException, resultException.code);
        ex.message = resultException.message;
        return ex;
    } else if (e instanceof JsonParseException
            || e instanceof JSONException
            || e instanceof ParseException) {
        ex = new ResponeThrowable(e, ERROR.PARSE_ERROR);
        ex.message = "解析错误";
        return ex;
    } else if (e instanceof ConnectException) {
        ex = new ResponeThrowable(e, ERROR.NETWORD_ERROR);
        ex.message = "连接失败";
        return ex;
    } else if (e instanceof javax.net.ssl.SSLHandshakeException) {
        ex = new ResponeThrowable(e, ERROR.SSL_ERROR);
        ex.message = "证书验证失败";
        return ex;
    }
    else {
        ex = new ResponeThrowable(e, ERROR.UNKNOWN);
        ex.message = "未知错误";
        return ex;
    }
}

/**
 * 约定异常
 */
class ERROR {
    /**
     * 未知错误
     */
    public static final int UNKNOWN = 1000;
    /**
     * 解析错误
     */
    public static final int PARSE_ERROR = 1001;
    /**
     * 网络错误
     */
    public static final int NETWORD_ERROR = 1002;
    /**
     * 协议出错
     */
    public static final int HTTP_ERROR = 1003;

    /**
     * 证书出错
     */
    public static final int SSL_ERROR = 1005;
}

public static class ResponeThrowable extends Exception {
    public int code;
    public String message;

    public ResponeThrowable(Throwable throwable, int code) {
        super(throwable);
        this.code = code;

    }
 }

 public class ServerException extends RuntimeException {
    public int code;
    public String message;
 }
}

接着可以在 BaseSubscriber<T>中处理异常拉

public abstract class BaseSubscriber<T> extends Subscriber<T> {

private Context context;


public BaseSubscriber(Context context) {
    this.context = context;
}

@Override
public void onError(Throwable e) {
    Log.e("Tamic", e.getMessage());
    // todo error somthing

    if(e instanceof ExceptionHandle.ResponeThrowable){
        onError((ExceptionHandle.ResponeThrowable)e);
    } else {
        onError(new ExceptionHandle.ResponeThrowable(e, ExceptionHandle.ERROR.UNKNOWN));
    }
 }
}

最后上层调用就是这样了:

 RetrofitClient.getInstance(MainActivity.this).createBaseApi().getData(new   BaseSubscriber<IpResult>(MainActivity.this) {

                @Override
                public void onError(ResponeThrowable e) {
                   // 处理翻译后异常。
                    Log.e("Tamic", e.code + " "+ e.message);
                    Toast.makeText(MainActivity.this, e.message, Toast.LENGTH_LONG).show();

                }

                @Override
                public void onNext(IpResult responseBody) {
                    Toast.makeText(MainActivity.this, responseBody.toString(), Toast.LENGTH_LONG).show();
                }
            }, "21.22.11.33");

值的注意的是上层使用BaseSubscriber 的实现类和子类即可,如果你想要重写BaseSubscriber 的onStat()和onCompleted() 也是可以的, 一般BaseSubscriber只处理公用的处理,或者进行下业务对返回格式检查,具体成功
解析有他的子类(实现类)去做。

注意:如果你不想将业务分发加到错误回调中,也可以这样做: 好比有的人喜欢将业务处理加到业务回调中,如果后台返回的业务码并不成功码的情况下, 不想走错误回调,也不想走成功回调, 想走直走业务回调。

可以这样处理:

在onNext() 中回调一个自定义的抽象的onBusiness(code, masg),将他的子类去实现

                       @Override
                        public void onNext(BaseResponse<IpResult> responseBody) {

                            if (!responseBody.isOk()) {
                                  //业务分发
                                 onBusiness(responseBody.getCode, responseBody.getMsg)
                            } else {
                                  // 成功回调
                                  onNext(responseBody.getData())
                            }

                        }

缓存问题

公共缓存
有时候需要在无网络时增加缓存功能,因此给Retrofit加入基础拦截器,来处理缓存问题

/**
    * BaseInterceptor
    * Created by Tamic on 2016-7-15.
    */
public class BaseInterceptor implements Interceptor{
    private Map<String, String> headers;
    private Context context;
    public BaseInterceptor(Map<String, String> headers, Context context) {
        this.headers = headers;
        this.context = context;
    }

    @Override
    public Response intercept(Chain chain) throws IOException {

        Request.Builder builder = chain.request()
                .newBuilder();
        builder.cacheControl(CacheControl.FORCE_CACHE).url(chain.request().url())
        .build();

        if (!NetworkUtil.isNetworkAvailable(context)) {

            ((Activity)context).runOnUiThread(new Runnable() {
                @Override
                public void run() {
                    Toast.makeText(context, "当前无网络!", Toast.LENGTH_SHORT).show();
                }
            });
        }
        if (headers != null && headers.size() > 0) {
            Set<String> keys = headers.keySet();
            for (String headerKey : keys) {
                builder.addHeader(headerKey, headers.get(headerKey)).build();
            }
        }

        if (NetworkUtil.isNetworkAvailable(context)) {
            int maxAge = 60; // read from cache for 60 s
            builder
                    .removeHeader("Pragma")
                    .addHeader("Cache-Control", "public, max-age=" + maxAge)
                    .build();
        } else {
            int maxStale = 60 * 60 * 24 * 14; // tolerate 2-weeks stale
            builder
                    .removeHeader("Pragma")
                    .addHeader("Cache-Control", "public, only-if-cached, max-stale=" + maxStale)
                    .build();
        }
        return chain.proceed(builder.build());

    }
}

okHttpClient加入拦截器

       okHttpClient = new OkHttpClient.Builder()
             .addInterceptor(new BaseInterceptor(headers))
            .addInterceptor(new   CaheInterceptor(context))
            .addNetworkInterceptor(new CaheInterceptor(context))
            .build();

Retrofit 加入okhttpClient

retrofit = new Retrofit.Builder()
                .client(okHttpClient)
                .baseUrl(url)
                .build();

单独缓存

如果你不想加入公共缓存,想单独对某个api进行缓存,可用Headers来实现,那么可以这样:

@Headers("Cache-Control : public, max-age = 3600")
@GET("service/getIpInfo.php")
Observable<BaseResponse<IpResult>> getData(@Query("ip") String ip);

值得注意的是 下面的两句话也必须加入:

             .addInterceptor(new   CaheInterceptor(context))
             .addNetworkInterceptor(new CaheInterceptor(context))

缓存路径和默认大小

如果想更改okhttp的缓存路劲,可以设置cache的path路径 ,姿势如下

 Cache cache = new Cache(httpCacheDirectory, 10 * 1024 * 1024);

第一个参数是路径,第二个最大缓存大小

                      okHttpClient = new OkHttpClient.Builder()
                          .cache(cache)
                           .build();

这样就加入自定义的Cache策略

自定义缓存

如果你不想用okhttp自带的缓存策略,因为这需要服务端配合处理缓存请求头,不然会抛出: HTTP 504 Unsatisfiable Request (only-if-cached)

除了以上修改 Request.cacheControl的方式实现缓存,也可以自定义一个Cahe策略用来实现本地硬缓存。

构建CaheManager,用Url对应Json实现,此类非常简单,你可以自己实现,时间策略可自我加入扩展
在BaseSubscriber进行网络判断,加载缓存数据返回妥妥的;

 @Override
 public void onStart() {
    super.onStart();

    Toast.makeText(context, "http is start", Toast.LENGTH_SHORT).show();

    // todo some common as show loadding  and check netWork is NetworkAvailable
    // if  NetworkAvailable no !   must to call onCompleted
    if (!NetworkUtil.isNetworkAvailable(context)) {
        Toast.makeText(context, "无网络", Toast.LENGTH_SHORT).show();

        if (isNeedCahe) {
            Toast.makeText(context, "无网络,已智能读取缓存!", Toast.LENGTH_SHORT).show();
            IpResult ipResult = new Gson().fromJson(CaheManager.getjson(url), IpResult.class);
            onNext((T) ipResult);
        }
        onCompleted();
    }

}

总结

通过这次的整理,再进行RxJava和Retrofit中 ,所有坑直接添就行,接着上次的介绍,笔者进行新框架开发novate已快接近尾声,估计本月就能和大家见面,敬请继续关注!

Retrofit 2.0系列请阅读

第一时间获取技术文章请关注扫码微信公众号!

开发者技术前线

源 码https://github.com/Tamicer/Novate

本文已在版权印备案,如需转载请访问版权印。66790509

相关文章

网友评论

  • 1a356b973f3a:public <T> Observable.Transformer<BaseResponse<T>, T> transformer() {

    return new Observable.Transformer() {

    @Override
    public Object call(Object observable) {
    return ((Observable) observable)
    .map(new HandleFuc<T>())
    .onErrorResumeNext(new HttpResponseFunc<T>());
    }
    };
    }

    private class HandleFuc<T> implements Func1<BaseResponse<T>, T> {
    @Override
    public T call(BaseResponse<T> response) {
    if (!response.isOk()) throw new RuntimeException(response.getCode() + "" + response.getMsg() != null ? response.getMsg(): "");
    return response.getData();
    }
    }

    大咖你好,最近在学习您的代码。但是用map转换数据类型调用HandleFuc()的时候总是报这个异常,麻烦帮忙看一下 多谢
    java.lang.ClassCastException: okhttp3.ResponseBody$1 cannot be cast to com.tamic.retrofitclient.net.BaseResponse
    at com.tamic.retrofitclient.net.RetrofitClient$HandleFuc.call(RetrofitClient.java:287)
  • bc1ff59c5b30:感觉楼主写代码还可以,但是写文章就不是很行了,呵呵
    Tamic:@chords :smiley:
  • 一大条咸鱼:我在onstart里面做了判断,但是它还是报java.net.ConnectException,尝试连接并失败了
  • bdpqchen:CaheInterceptor 少个c
  • 离氺的鱼:rx2中怎么统一处理基础Subscriber??2里面好像和1完全的不同,求个思路!!
  • 3085ba51ad91:大神数据存储在哪啊,只是看见了拦截器。
  • 有兴不虚昧: showLoadingProgress();放在onStart()里面不好吧,这个要放在主线程里面的
    Tamic: @有兴不虚昧 是的
  • MonkiRayman:mark一波
  • shixinBook:每当自己学得差不多的时候,突然之间看到了别人的博客,才发现自己真是够菜。
    Tamic:@shixinBook 互勉
  • 流水不腐小夏:只在complete中调用closeLoadingProgress,那么error的情况下呢
    Tamic:@流水不腐小夏 也需要关掉的
  • 赛冷死:范磊...我认识你~~~
    Tamic:@赛冷死 不会吧 哈哈
    赛冷死:@Tamic 一个分享你文章的人,没细看还以为是他写的这篇~尴尬了~:relaxed:
    Tamic:范磊是谁?
  • 布鲁马:在这里写感谢作者无私的开源。借我参考参考。 :blush:
    缓存策略感觉自己加的跟翔一般,网络框架写的太死板了,没有任何扩展性

    再次感谢码主!!
    Tamic:@布鲁马 欢迎交流
  • de89be2fcf9a: HTTP 504 Unsatisfiable Request (only-if-cached) 大神对于这个问题怎么处理的
    de89be2fcf9a:@Tamic 服务器是不可能配合的 这辈子都不可能了 看来要自定义缓存策略了
    Tamic:@Tamic request = request.newBuilder()
    .cacheControl(CacheControl.FORCE_CACHE)
    .build();
    加上这句试试
    Tamic:@刘婧琳 服务器要配合的
  • 一盏孤灯守着孤城:还想问下,这个缓存 是写到本地的吧。为什么我退出app后,再进来就没有缓存了。
    Tamic:@日你去 是的
    一盏孤灯守着孤城:@Tamic 我想问下 我有网的时候debug看到CacheInterceptor里面有网的回调。但是无网的时候debug不这个回调 也没有 toast本地智能加载。
    我用的是这种 缓存 @Headers("Cache-Control: public, max-age=3600") 加头的方式,你意思是说服务器那边要处理?
    Tamic:@日你去 是本地的
  • 一盏孤灯守着孤城:大神 我想问个问题。那我怎么在缓存中删除我不想要的了。比如说我的页面有删除功能。我刷新的时候没有网络。然后走的就是缓存。但是我页面操作的删除 ,删除不了缓存里面的内容。这个怎么弄。
    Tamic: @一盏孤灯守着孤城 如果你想对同一个接口的数据单条进行删除 要么自己实现一个缓存控制类 要么借助RXCache去实现
    Tamic: @一盏孤灯守着孤城 看到比较晚 可以拿到okhttp的缓存控制类clear一下
  • 一盏孤灯守着孤城:我想问下,我这是做单个api的缓存。为什么CaheInterceptor 没有这个类
    我用的也是 compile 'com.squareup.retrofit2:retrofit:2.0.0-beta4'
    Tamic:@日你去 是的 修改请求头
    一盏孤灯守着孤城:@日你去 难道还要自己写实现Interceptor 的实现类CaheInterceptor 么
    Tamic:@日你去 自己命名的拦截器
  • 5a0d7d7a9b4f:楼主可以提供一下文章末尾所说的CacheManager吗,还有可以问一下在onStart里面怎么获取到url
    Tamic: @潇潇的鱼儿 切分之1.x
    46b1434d316d:@Tamic 楼主的开源项目链接,怎么没有依赖源码。。。
    Tamic:@saky0542自己写一个 map<url, jsonString> 映射一下就行了
  • zhuhf:onStart(): 这是 Subscriber 增加的方法。它会在 subscribe 刚开始,而事件还未发送之前被调用,可以用于做一些准备工作,例如数据的清零或重置。这是一个可选方法,默认情况下它的实现为空。需要注意的是,如果对准备工作的线程有要求(例如弹出一个显示进度的对话框,这必须在主线程执行), onStart() 就不适用了,因为它总是在 subscribe 所发生的线程被调用,而不能指定线程。要在指定的线程来做准备工作,可以使用 doOnSubscribe() 方法,具体可以在后面的文中看到。
    Tamic:@hiphonezhu 感谢你提供的好的建议 我后期再优化优化
  • f11f741d48b4:请问 BaseSubscriber 中 的 url 是怎么拿到的呢?
    Jessewo:@Tamic 那么请问 interceptor是如何将url 传给BaseSubscriber 的?:sweat:
    Tamic:@lllHa1o 通过拦截器可以拿到的
  • 8c89095d593f:和我们项目封装的有点像,但思路不同可以借鉴。我这是封装了obserble,只有一个post和get,但造成的问题是自己得解析json,全部返回responsebody,这些和错误信息以及数据库缓存都写在httphelper,调用接口全走helper。至于用数据库是因为okhttp缓存拿请求url作为md5当作key,但同时带来的问题就是我的请求由于有token加密,同一个接口就会有多个key,缓存的数据就没有意义了。
    Tamic:@流清 如果需要缓存那么请将token放到cookie里面,这样Url还是一致的,如果放到参数里面的,也可以自定实现一个缓存机制,用你自己定义的keyd对应数据库里的一串json。
    8c89095d593f:@Tamic_码小白 token里面是根据当前时间来做加密的,就是说url是一直在变的,这种怎么配置呢
    Tamic:@流清 确实Url一变化,okhttp就会默认新的缓存,所以对于加密数据你缓存设置编码格式也是可以的
  • Jafir:你好 问题解决了 是没有使用subscriber或者observer,用的action没有处理onError出现的问题。
    Jafir:@三刀you三刀 就是你subscribe最好是一个subscriber而不是action1 因为 action1不能处理异常情况
    ec15b7666993:请问能说详细点吗?我也遇到这样的问题,求解决方法
    Tamic:@Jafir :+1:
  • Jafir:你好,我遇到一个问题就是 如果没有网络 并且又没有缓存的话,就会报错 java.lang.IllegalStateException: Exception thrown on Scheduler.Worker thread. Add `onError` handling.
    rx.exceptions.OnErrorNotImplementedException: HTTP 504 Unsatisfiable Request
    意思就是rxjava没有捕获到这个异常,并且,这个异常在scheduler worker线程里面
    请问怎么解决呢?
    Tamic:@Jafir 自己做判断
    Tamic:@Jafir 有调用的代码吗?
  • 24K纯帅豆:请问一下,请求的时候报这个错,楼主知道是什么情况吗?
    java.lang.IllegalStateException: Fatal Exception thrown on Scheduler.Worker thread.
    Tamic:@24K纯帅豆 你重新下载工程 我今天更新了细致的错误异常处理
    24K纯帅豆:@Tamic_码小白 .subscribeOn(Schedulers.io())
    .unsubscribeOn(Schedulers.io())
    .observeOn(AndroidSchedulers.mainThread())
    .subscribe(subscriber);线程切换应该是没有错的,返回的数据参数也是按照返回的json串来命名的
    Tamic:@24K纯帅豆 线程切换错了吧 或者你返回数据参数出错了 就直接报Scheduler.Worker异常了
  • 68768b474bfc:能不能解释一下缓存的原理
    Tamic:@Tamic 已经更新了一小点 你可以慢慢看看下面的
    68768b474bfc:@Tamic_码小白 期待
    Tamic:@TellH 下一篇来个补充篇 和http协议有关
  • KunMinX:我十分不解的是,为什么网络编程有这么多花样要人们具体去实现。这不都是千篇一律的吗,这不应该封装好直接拿来用无需知道其中细节的吗。而且居然可以上升到框架的程度。网络编程真的有那么复杂吗,不同的需求,细节真的大不相同吗。

    我理想中的网络编程,它是一个库,对应发送与接收,它只暴露两个方法给我:
    sendData(string path,object data)
    getData(string)

    从结果上看,难道不就是这么简单吗
    Hi_Felix:如果你要处理一下第一次请求返回的数据,再请求第二次,你该怎么做呢?或者你要上传图片,上传完毕之后保存最新的图片状态,该怎么做呢?两个方法就搞定了?你是想的太天真了还是根本没做过稍微大点的项目。
    Mansoul:还有大多情况下我们需要对request和response做自己的处理,这个只能你自己去实现了
    Tamic:@简洁笔记 其实约单一 扩展能力就越差 所以一般框架都会给上层给扩展能力的
  • 薄炳鑫:关注持续
  • bcf47d7497d7:请问,retrofit现在在企业里面用的多吗?在数据传说这一块处于什么地位呢,蟹蟹了,刚接触,不太清楚
    Tamic:@KevinDakui 普及率很高了已经
  • starCoder:期待新框架的到来,到时借鉴借鉴经验
    starCoder:@Tamic_码小白 研究下
    Tamic:@star_code https://github.com/NeglectedByBoss/Novate/
    Tamic:@star_code 谢谢支持

本文标题:Rxjava +Retrofit 你需要掌握的几个技巧,Retr

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