摘要:因为使用默认将返回的对象对象转换成了的对象了使用但是我们在使用后,有时可能会这样声明接口这样定义可以很方便的在返回中直接一个对象了登录成功登录失败但是这样有一个问题就是会丢失的元数据,因为对象我们是没法访问的。
Retrofit:声明 ApiService接口
我们在使用Retrofit的时候只需要把URL通过注解的形式写到APIService文件中就行了。
比如登录功能:
如果后台的成功返回格式为
{ code:0; Message:"login success" }
失败的返回格式为
{ code:-1; Message:"login failed" }
我们定义个一个Bean类LoginResult.
public class LoginResult{ private int code; private String message; //get/set }不使用Rxjava
在方法的定义中,如果不适用Rxjava的话,
@Post @FormUrlEncoded Calllogin(@Field("userName") String userName, @Field("password")String password);
在使用的时候们只需要使用apiService.login("dsf", "dsf"),就好了。该方法会返回一个Call
我们只需要或者异步执行
//同步执行 Responseresponse = call.execute(); //异步调用 call.enqueue(new Callback() { @Override public void onFailure(Call call, IOException e) { } @Override public void onResponse(Call call, Response response) throws IOException { } });
然后我们在返回结果里面操作就行了,如果我们需要Response里面的元数据,比如header里面的信息的话,直接从Response取就好了。
因为Retrofit 使用DefaultCallAdapterFactory默认将OKHttp返回的Call对象(OkHttp对象)转换成了Retrofit的Call
但是我们在使用RxJava后,有时可能会这样声明接口
@Post @FormUrlEncoded Observablelogin(@Field("userName") String userName, @Field("password")String password);
这样定义可以很方便的在返回中直接subScribe一个Action对象了
Observableobservable = model.login("dfs", "dfs"); observable.subscribe(new Action1 () { @Override public void call(LoginResult loginResult) { if (loginResult.getCode() == 0) { //登录成功 } else { //登录失败 } } });
但是这样有一个问题就是会丢失Response的元数据,因为Response对象我们是没法访问的。因为Retrofit已经帮助我们做了转换,直接把我们接口方法里定义的数据类型转换好后返回给我们了。去掉了Response信息。
Observable我们可以这么定义接口
@Post @FormUrlEncoded Observable> login(@Field("userName") String userName, @Field("password")String password);
这样我们就得到这样的返回结果。
Observable> observable = model.login("dfs", "dfs"); observable.subscribe(new Action1 >() { @Override public void call(Response loginResultResponse) { if (loginResultResponse.code() == 200){ LoginResult body = loginResultResponse.body(); }else { //登录失败 } } });
这样就能拿到Response信息了,然后根据Response code判断是否登录成功了。
但是这样有个问题是我们写接口会特备繁琐,每次都得使用Response<>泛型。一般我们都写作Observable
其实我们一般情况下也不关注Response信息的。但是不排除特殊情况
比如这种情况
https://academy.realm.io/cn/p...
这篇文章提到的分页加载的情况,其实这种情况也可以将下一个页面的URL放到Response body里返回。
这里只是给出了一种情景。
现实中解决方案很多种,只能折中了,其实我们没必要使用Response的Response code 来判断接口是否调用成功了。因为Retrofit都帮我们做过了。所以我们定义接口的时候只要使用Observable
我们在使用Retrofit + Rxjava 的时候一般都这么生成Retrofit client的
Retrofit retrofit = new Retrofit.Builder() .client(okHttpClient) .addConverterFactory(GsonConverterFactory.create()) .addCallAdapterFactory(RxJavaCallAdapterFactory.create()) .baseUrl(baseUrl) .build();
这里RxJavaCallAdapterFactory生成的CallAdapter对象就帮我们做好了结果的转换
我们看一下RxJavaCallAdapterFactory的声明。
/** * A {@linkplain CallAdapter.Factory call adapter} which uses RxJava for creating observables. ** Adding this class to {@link Retrofit} allows you to return {@link Observable} from service * methods. *
* There are three configurations supported for the {@code Observable} type parameter: 有三种配置支持Observable的类型参数 ** interface MyService { * @GET("user/me") * Observable
getUser() * } *
结论就是,如果我们没有对Response有特殊的需求的话,直接在接口声明处直接声明成Observable
这篇文章本应该和Retofit源码分析一起写的,奈何Retrofit的源码实在是太晦涩难懂了
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/75933.html
摘要:看下图所示,摘自网络的创建流程源码分析实例是使用建造者模式通过类进行创建的。创建了一个含有对象实例的,并返回给源码分析添加一个调用适配器工厂,用于支持服务方法返回类型注意生产的是,那么又是什么呢可以看到源代码如下所示,它是一个接口。 目录介绍 1.首先回顾Retrofit简单使用方法 2.Retrofit的创建流程源码分析 2.1 Retrofit对象调用Builder()源码解...
阅读 3094·2021-09-24 10:26
阅读 3269·2021-09-23 11:54
阅读 4687·2021-09-22 15:33
阅读 2252·2021-09-09 09:33
阅读 1655·2021-09-07 10:10
阅读 959·2019-08-30 11:09
阅读 2848·2019-08-29 17:13
阅读 1007·2019-08-29 12:35