美文网首页Android大话安卓Rxjava2 讲解系列
Rxjava、Retrofit返回json数据解析异常处理

Rxjava、Retrofit返回json数据解析异常处理

作者: 莫比乌丝环丶 | 来源:发表于2017-02-20 16:41 被阅读8175次

每个App都避免不了要进行网络请求,从最开始的用谷歌封装的volley到再到android-async-http再到OKHttpUtils再到现在的RetrofitRxJava,从我自己用后的体验来看,用了retrofit和RxJava真的回不去了,回不去了,不去了,去了,了...(哈哈,本来还想分析下这四个的区别,网上这样的文章很多,我就没必要多添乱了-.-)。不多逼逼,下面开始正文。

RxJava的学习资料

RxJava javadoc

ReactiveX文档中文翻译

1、Rxjava和Retrofit依赖导入:

compile 'io.reactivex:rxandroid:1.2.0'                       //Rxjava专门针对anroid封装的RxAndroid
compile 'io.reactivex:rxjava:1.1.5'
compile 'com.squareup.retrofit2:retrofit:2.0.2' //retrofit
compile 'com.squareup.retrofit2:converter-gson:2.0.2' //gson converter
compile 'com.squareup.retrofit2:adapter-rxjava:2.0.2' //Retrofit专门为Rxjava封装的适配器
compile 'com.google.code.gson:gson:2.6.2' //Gson
compile 'com.squareup.okhttp3:logging-interceptor:3.1.2' //打印网络请求的log日志

2、网络请求基类的配置

建立一个工厂类

public class ServiceFactory {

private final Gson mGsonDateFormat;


public ServiceFactory(){
mGsonDateFormat = new GsonBuilder()
.setDateFormat("yyyy-MM-dd hh:mm:ss")
.create();
}


private static class SingletonHolder{
private static final ServiceFactory INSTANCE = new ServiceFactory();
}


public static ServiceFactory getInstance(){
return SingletonHolder.INSTANCE;
}


public <S> S createService(Class<S> serviceClass){
Retrofit retrofit = new Retrofit.Builder()
.baseUrl(Constant.BASE_URL) //Retrofit2 base url 必须是这种格式的:http://xxx.xxx/
.client(getOkHttpClient())

--------------------------添加Gson工厂变换器也就是不用管数据解析-----------------------------------
.addConverterFactory(GsonConverterFactory.create())
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
.build();
return retrofit.create(serviceClass);

}

HttpLoggingInterceptor loggingInterceptor = new HttpLoggingInterceptor(new HttpLoggingInterceptor.Logger() {
@Override
public void log(String message) {
//打印retrofit日志
Log.i("RetrofitLog","retrofitBack ======================= "+message);
}
});


private static final long DEFAULT_TIMEOUT = 10;
private OkHttpClient getOkHttpClient(){
loggingInterceptor.setLevel(HttpLoggingInterceptor.Level.BODY);
//定制OkHttp
OkHttpClient.Builder builder = new OkHttpClient.Builder();
builder.connectTimeout(DEFAULT_TIMEOUT, TimeUnit.SECONDS);
builder.writeTimeout(DEFAULT_TIMEOUT,TimeUnit.SECONDS);
builder.addInterceptor(loggingInterceptor);
//设置缓存
File httpCacheDirectory = new File(SDCardUtils.getRootDirectoryPath(),"这里是你的网络缓存存放的地址");
builder.cache(new Cache(httpCacheDirectory,10*1024*1024));

return builder.build();
}

}

好了 下一步 我们要建一个网络请求的基类,一般网络请求返回的数据最外层的根式就是 code msg result,可变的就是result,所以我们把result的类型定义为一个泛型的

public class HttpResult<T> extends BaseEntity {
public int code;
private boolean isSuccess;
private T result;
private String msg;

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


public T getResult() {
return result;
}


public void setResult(T result) {
this.result = result;

}

public boolean isSuccess() {
return code == 200;
}

public int getCode() {
return code;
}
}

既然我们的网络请求是rxjava配合retrofit 下面就定义我们的网络请求订阅subscriber

public abstract class HttpResultSubscriber<T> extends Subscriber<HttpResult<T>> {
@Override
public void onNext(HttpResult<T> t) {
if (t.isSuccess()) {
onSuccess(t.getResult());
} else {
_onError(t.getMsg().getCode());
}
}

@Override
public void onCompleted() {

}

@Override
public void onError(Throwable e) {
e.printStackTrace();
//在这里做全局的错误处理
if (e instanceof ConnectException ||
e instanceof SocketTimeoutException ||
e instanceof TimeoutException) {
//网络错误
_onError(-9999);
}
}


public abstract void onSuccess(T t);

public abstract void _onError(int status);
}

OK我们的网络请求基类已经完成啦!下面开始我们的网络请求

首先我们定义一个接口(以登录为例):

public interface LoginService {

//这个例子是post为例,如果想要了解其他的网络请求,请点击文章开始出的retrofit链接
@FormUrlEncoded
@POST(Constant.LOGIN_URL) 这里是你的登录url

//可以看到我们的登录返回的是一个Observable,它里面包含的使我们的网络请求返回的实体基类,

//而我们实体基类的result现在就是UserInfoEntity
Observable<HttpResult<UserInfoEntity>> login(@Field("mobile") String phone,
@Field("password") String pwd);
}

现在开始我们的网络请求啦

public void login(String phone, String pwd) {
ServiceFactory.getInstance()
.createService(LoginService.class)
.login(phone,pwd)
.compose(TransformUtils.<HttpResult<UserInfoEntity>>defaultSchedulers())
.subscribe(new HttpResultSubscriber<UserInfoEntity>() {
@Override
public void onSuccess(UserInfoEntity userInfoEntity) {
//这是网络请求陈宫的回调
}

@Override
public void _onError(int status) {
//这是失败的回调 根据status做具体的操作
}
});
}

你以为这样就行了 ,  这样子确实没毛病,确实已经妥妥的了。可是,可是,事与愿违啊!!!

3、具体解决办法

一般情况这是我们的返回json格式:

{
"code":200,
"msg":"成功",
"result":{}
}

我们刚才定义登录接口的时候 返回的实体基类例传入的是UserInfoEntity  这确实是没问题的 可是你们加入登录失败的时候返回的json数据格式是这样的怎么办?

{
"code":300,
"msg":"成功",
"result":[]
}

失败的时候返回的实体又是一个数组,这样子就会抱一个json解析异常拿不多失败的状态码和提示信息

OK其实我们的网络请求已经完成90%了,剩下的就是不重要的失败的时候回调了。

方法一:(这是在后台兄弟好说话,而且不打人的情况下...当然这种好人,还是有的,不过这不是我们今天要讲的重点)

我们可以让后台返回的json数据中的result永远是个数组。

{
"code":300,
"msg":"成功",
"result":[]
}

方法二:

        首先给大家看一个图片

这就是我们要下手的地方

上面我们添加的工厂变换器是导入的依赖 compile 'com.squareup.retrofit2:converter-gson:2.0.2' 这个提供的,

那可能有人要问了,那我们不用这个用哪个啊,不着急,不着急。还好retrofit是支持自定义的ConverterFactory的

下面我们就开始我们的自定义征程吧。

---------------------------------------------------------华丽丽的分割线-----------------------------------------------

1、自定义Gson响应体变换器


public class GsonResponseBodyConverter<T> implements Converter<ResponseBody,T>{
private final Gson gson;
private final Type type;


public GsonResponseBodyConverter(Gson gson,Type type){
this.gson = gson;
this.type = type;
}
@Override
public T convert(ResponseBody value) throws IOException {

String response = value.string();
//先将返回的json数据解析到Response中,如果code==200,则解析到我们的实体基类中,否则抛异常
Response httpResult = gson.fromJson(response, Response.class);
if (httpResult.getCode()==200){
//200的时候就直接解析,不可能出现解析异常。因为我们实体基类中传入的泛型,就是数据成功时候的格式
return gson.fromJson(response,type);
}else {
ErrorResponse errorResponse = gson.fromJson(response,ErrorResponse.class);
//抛一个自定义ResultException 传入失败时候的状态码,和信息
throw new ResultException(errorResponse.getCode(),errorResponse.getMsg());
}
}
}

2、自定义一个响应变换工厂 继承自 retrofit的 converter.Factory

public class ResponseConverterFactory extends Converter.Factory {

public static ResponseConverterFactory create() {
return create(new Gson());
}


public static ResponseConverterFactory create(Gson gson) {
return new ResponseConverterFactory(gson);
}

private final Gson gson;

private ResponseConverterFactory(Gson gson) {
if (gson == null) throw new NullPointerException("gson == null");
this.gson = gson;
}

@Override
public Converter<ResponseBody, ?> responseBodyConverter(Type type, Annotation[] annotations, Retrofit retrofit) {
//返回我们自定义的Gson响应体变换器
return new GsonResponseBodyConverter<>(gson, type);
}

@Override
public Converter<?, RequestBody> requestBodyConverter(Type type,
Annotation[] parameterAnnotations, Annotation[] methodAnnotations, Retrofit retrofit) {
//返回我们自定义的Gson响应体变换器
return new GsonResponseBodyConverter<>(gson,type);
}
}
//然后将上面的GsonConverterFactory.create()替换成我们自定义的ResponseConverterFactory.create()

然后将上面的GsonConverterFactory.create() 替换成我们自定义的ResponseConverterFactory.create()。

public <S> S createService(Class<S> serviceClass){
Retrofit retrofit = new Retrofit.Builder()
.baseUrl(Constant.BASE_URL)
.client(getOkHttpClient())
//.addConverterFactory(GsonConverterFactory.create())
//然后将上面的GsonConverterFactory.create()替换成我们自定义的ResponseConverterFactory.create()
.addConverterFactory(ResponseConverterFactory.create())
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
.build();
return retrofit.create(serviceClass);

}

再然后,最后一个然后啦(-.-)

在我们的自定义的Rxjava订阅者 subscriber中的onError()中加入我们刚才定义的ResultException。

@Override
public void onError(Throwable e) {
e.printStackTrace();
//在这里做全局的错误处理
if (e instanceof ConnectException ||
e instanceof SocketTimeoutException ||
e instanceof TimeoutException) {
//网络错误
_onError(-9999);
} else if (e instanceof ResultException) {
//自定义的ResultException
//由于返回200,300返回格式不统一的问题,自定义GsonResponseBodyConverter凡是300的直接抛异常
_onError(((ResultException) e).getErrCode());
System.out.println("---------errorCode------->"+((ResultException) e).getErrCode());
}
}

这次是真的完成了我们的json数据解析异常的处理,其实我们的解决办法是解析了两次,第一次解析的时候我们的Response中只有只是解析了最外层的 code 和 msg  ,result中的是没有解析的。response中的code==200,直接将数据解析到我们的实体基类中。如果code!=200时,直接抛自定义的异常,直接会回调到subscriber中的onError()中。虽然进行了两次解析,但是第一次只是解析了code,和msg 对于效率的影响其实并不大,在功能实现的基础上一点点效率的影响(而且这个影响是微乎其微的-.-)其实无伤大雅的。

相关文章

  • Rxjava、Retrofit返回json数据解析异常处理

    每个App都避免不了要进行网络请求,从最开始的用谷歌封装的volley到再到android-async-http再...

  • Retrofit + Gson解析数据容错处理

    使用Retrofit+Gson 处理网络数据,后台数据返回不规范,因为种种原因后台无法修改,直接解析会报错Json...

  • Retrofit 2.0 笔记

    简单的用法 解析 JSON 自定义解析方法 RxJava + Retrofit 参考:RetrofitAndroi...

  • Json学习

    json的返回与解析 添加json依赖包以及处理json为bean的包 返回json格式数据到前端(返回信息均为键...

  • 项目开发

    Retrofit封装 RxJava 与 Retrofit 结合的最佳实践 flatmap统一处理异常 Rx处理服务...

  • android protobuf ProtoConverter

    proobuf 无法解析问题!!!!!!!!! 最近使用Retrofit+Rxjava 解析proobuf 数据流...

  • Expected BEGIN_OBJECT but was BE

    在使用Rxjava+Retrofit进行网络请求获取数据后,通过Gson解析会报这个错: 原因是返回的数据是一个数...

  • 王学岗gson解析出错

    解析json一直报错,这个异常Expected ?but was ?,查看了下返回的数据格式,发现服务器返回的数据...

  • RXJava compose操作符消除重复代码

    信小伙伴在使用RXJava与Retrofit请求网络时,都有遇到过这样的场景,在IO线程请求网络解析数据,接着返回...

  • 用Kotlin的方式来处理网络异常

    一. 前言 之前的文章 RxJava处理业务异常的几种方式 曾经介绍过 Retrofit 的异常可以有多种处理方式...

网友评论

  • 154db6c439ab:我今天遇到了一模一样的例子,请问楼主这个有demo吗?这里只是部分的代码,有些类找不到的,有demo麻烦发一下,谢谢!
    莫比乌丝环丶:@小二木 欢迎加入大话安卓,群聊号码:271127803
  • 3670c31b7212:请问有demo参考吗
  • a682b8a9bb06:感谢楼主,这个后台返回数据坑死我了:fearful:
  • f6918f4bdaf2:您好,我遇到携带参数会报错的问题,该怎么处理
    错误信息:FeedbackRequest cannot be cast to okhttp3.ResponseBody
    请求接口:Observable<HttpResult<Object>> feedback(@Body FeedbackRequest request);
    f6918f4bdaf2:@Override
    public Converter<?, RequestBody> requestBodyConverter(Type type,
    Annotation[] parameterAnnotations, Annotation[] methodAnnotations, Retrofit retrofit) {
    //返回我们自定义的Gson响应体变换器
    TypeAdapter<?> adapter = gson.getAdapter(TypeToken.get(type));
    return new GsonRequestBodyConverter<>(gson, adapter);
    }
    你这个方法不能返回new GsonResponseBodyConverter<>(gson, type) ,这个请求参数,不能用你那个Converter 应改为以上代码,我是根据GsonConverterFactory源码写的
  • 蘑菇碳烤小鸡排:我这边的情况是不管成功与否code值是不变的,只是data的结构变了
    莫比乌丝环丶:@蘑菇碳烤小鸡排 应该也可以吧。只是判断有点麻烦而已。我个人觉得应该你们后台改一下。从没见过这么来的
  • 白昭羽:按照楼主的写了,但是请求成功的时候服务端返回的数据有时候是"data":{}有时候是"data":[],在data的时候仍旧解析失败了,报了Expected BEGIN_ARRAY but was BEGIN_OBJECT这个错
    3670c31b7212:@莫比乌丝环丶 请问群是多少?
    白昭羽:@莫比乌丝环丶 好的,谢谢
    莫比乌丝环丶:@白昭羽 网络回调的基类,data 属性是一个泛形,成功的回调类型你是知道的。出现你说的错误应该是你写的有问题。你在看看我的代码。如果还有问题你加群问我
  • da2245eebc1d:我们后台说了,就是code != 200的情况下 ,data里也有可能有数据让我们解析,何解?
    莫比乌丝环丶:@冰之AJ 我这里错误的信息定义了实体类,你可以把把里面的属性换成你们后来的即可
  • 张超_862e:有完整的吗 差了好多东西,给发一份完整的学习一下
  • 410fd86a74cd:这样处理了之后,后台那边能拿到请求记录吗?
    莫比乌丝环丶:@小赵VS老付 这个是处理后台返回,数据错误合适不统一的问题。
  • 10e8a366f9e5:是不是既能解析data是{} 又能解析data是数组[]
    '{
    "code":200,
    "msg":"成功",
    "data":{}
    }'
    '{
    "code":300,
    "msg":"成功",
    "data":[]
    }'
    需要一个这样的ConverterFactory
    莫比乌丝环丶:@CYMOK 你永远知道成功返回的数据类型 data的数据类型你可以写成泛型T就OK了 。 此文是为了应对后台失败的时候返回的数据结构不统一的问题。
    莫比乌丝环丶:@CYMOK 是的
  • JieeOS:我这边后台返回给我的json格式是个数组格式 [{id : 1, name: st},````{}]
    用Retrofit的Gson能走onResponse的方法,想问问response.body()是返回什么呢
  • 2f16617a6130:很好很详细
  • 2465d6d59914:一直在百度上问这个问题,一直没得到解答,因为根本不知道怎么问这个问题,今天楼主这篇真是太棒了。完美解决问题
    莫比乌丝环丶: @2465d6d59914 😁😁😁
  • Super_l1:楼主你好,给个demo地址呀。初学者自己不好改
    莫比乌丝环丶: @AvALand 你加下这个群,然后私聊我,我发你完整的。271127803
  • 简单爱呵呵:灰常感谢,可以解决问题:+1:
    莫比乌丝环丶: @简单爱呵呵 有个类好像没上传
  • 嘿张开心:你这差好多类啊 不完整啊
    嘿张开心:太操蛋了
  • 过客Diane:有详细的封装么
    过客Diane:@莫比乌丝环丶 github地址有么,最近在准备用Rxjava和Retrofit组合请求应用到项目,刚好看到你这有个封装想借鉴一下
    莫比乌丝环丶: @过客Diane 有的
  • 3c65110f9123:要是请求不管成功与否,服务端返回的数据有时候是"data":{}有时候是"data":[],这种情况怎么办,可以解决吗
    莫比乌丝环丶: @0b9790fa85f0 没懂你的思路
    0b9790fa85f0:直接写一个返回String的解析工厂 就搞定了
    莫比乌丝环丶: @_随想 我的这篇文章就是解决这个问题的

本文标题:Rxjava、Retrofit返回json数据解析异常处理

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