RxJava(五) onErrorResumeNext操作符实现app与服务器间token机制
欢迎转载,转载请标明出处:
http://blog.csdn.net/johnny901114/article/details/51533586
本文出自:【余志强的博客】
一、需求场景:
在开发App的时候, 很多公司的提api接口, 请求的的时候都需要带有token, 该token在用户第一次启动app或者登陆的时候去获取. 以后的所有请求都需要带该Token 如果token过期, 服务器将返回401, 这时候就需要去请求获取token的接口, 如果获取成功接着在请求原来的接口. 这个时候就两个回调的嵌套了. 实现起来比较费劲, 而且也不够优雅. 代码的可维护性变得很差. 可以使用 onErrorResumeNext
来处理这样的业务逻辑.
例如:请求一个用户信息接口,如果token没有过期,返回用户信息,如果token过期,服务器返回401,客户端发一个获取新token的请求,成功后,再去请求用户信息接口。
二、如何使用onErrorResumeNext解决
使用Retrofit来访问服务器
private static RestAdapter restAdapter = new RestAdapter
.Builder()
.setLogLevel(RestAdapter.LogLevel.FULL)
.setEndpoint(BASE_URL)
.setErrorHandler(new NetWorkErrorHandler())
.setRequestInterceptor(requestInterceptor)
.build();
public static <S> S createService(Class<S> serviceClazz) {
return restAdapter.create(serviceClazz);
}
如果服务器返回401,我们要去请求新的token,下面来判断错误类型:
NetWorkErrorHandler
private static class NetWorkErrorHandler implements ErrorHandler {
@Override
public Throwable handleError(RetrofitError error) {
retrofit.client.Response r = error.getResponse();
if (r != null && r.getStatus() == 401) {
Log.e("ErrorHandler", "---------> access deny code=401");
// User Custom Exception
return new AccessDenyException(error.getMessage());
}
return error.getCause();
}
}
UserApi
public interface UserApi {
@GET("/token")
AuthToken refreshToken();
}
服务器端代码逻辑
服务器端使用Java web+Tomcat来实现的. 如果需要可以把服务器部署在你的本地机器上, github地址
服务器端的基本逻辑:客户端请服务器api,服务器判断客户端带过来的token,如果过期则返回401,提示没有权限访问;如果是请求token接口,则返回token,有效期为10s。
客户端App的实现
以一个请求用户信息接口为例
Observable<Response> observable = userApi.getUserInfo();
observable.onErrorResumeNext(refreshTokenAndRetry(observable))//also use retryWhen to implement it
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Observer<Response>() {
@Override
public void onCompleted() {
loading = false;
appendText(tvLogs, "task completed-----");
//hideLoadingDialog();
}
@Override
public void onError(Throwable t) {
//hideLoadingDialog();
t.printStackTrace();
loading = false;
appendText(tvLogs, t.getClass().getName() + "\n" + t.getMessage());
NetErrorType.ErrorType error = NetErrorType.getErrorType(t);
appendText(tvLogs, error.msg);
}
public void onNext(Response response) {
String content = new String(((TypedByteArray) response.getBody()).getBytes());
appendText(tvLogs, "receiver data: " + content);
}
});
核心代码
private <T> Func1<Throwable, ? extends Observable<? extends T>> refreshTokenAndRetry(final Observable<T> toBeResumed) {
return new Func1<Throwable, Observable<? extends T>>() {
@Override
public Observable<? extends T> call(Throwable throwable) {
throwable.printStackTrace();
// Here check if the error thrown really is a 401
if (isHttp401Error(throwable)) {
return createTokenObvervable().flatMap(new Func1<AuthToken, Observable<? extends T>>() {
@Override
public Observable<? extends T> call(AuthToken token) {
appendText(tvLogs, "refresh token success,token's validity is 10s\nResume last request");
return toBeResumed;
}
});
}
// re-throw this error because it's not recoverable from here
return Observable.error(throwable);
}
public boolean isHttp401Error(Throwable throwable) {
return throwable instanceof AccessDenyException;
}
};
}
请求token api 的Observable
public Observable<AuthToken> createTokenObvervable() {
return Observable.create(new Observable.OnSubscribe<AuthToken>() {
@Override
public void call(Subscriber<? super AuthToken> observer) {
try {
if (!observer.isUnsubscribed()) {
appendText(tvLogs, "God!!! Token is out of date. \nstart refresh token......");
observer.onNext(userApi.refreshToken());
observer.onCompleted();
}
} catch (Exception e) {
observer.onError(e);
}
}
}).subscribeOn(Schedulers.io());
}
运行效果:
RxJava(五) onErrorResumeNext操作符实现app与服务器间token机制的更多相关文章
- RxJava(七) 使用debounce操作符 优化app搜索功能
欢迎转载,转载请标明出处: http://blog.csdn.net/johnny901114/article/details/51555203 本文出自:[余志强的博客] 一.抛出问题 现在几乎所有 ...
- RxJava学习笔记(操作符)
前言 上一篇文章介绍了RxJava的基础知识和简单实现,篇幅已经比较多了,所以把操作符(Operators)相关的内容放在这一篇.有了上一篇文章的基础,相信会比较容易理解操作符相关的内容了. 操作符( ...
- app与服务器对接
如何做ios的app与服务器的数据传输
- 深入浅出RxJava(二:操作符)
看完这篇blog,我相信你肯定想立即在你的项目中使用RxJava了,这篇blog将介绍许多RxJava中的操作符,RxJava的强大性就来自于它所定义的操作符. 首先先看一个例子: 准备工作 假设我有 ...
- Mui --- app与服务器之间的交互原理、mui ajax使用
1.APP与服务器之间的交互原理 app端(客户端)与服务端的交互其实理解起来和容易,客户端想服务器端发送请求,服务器端进行数据运算后返回最终结果.结果可以是多种格式: 1.text 文本格式 2.x ...
- 如何把App放在服务器上供用户下载
如何把App放在服务器上供用户下载 有时候做了个简单的App想把App给朋友帮忙测试一下,却发现上传到各种平台很麻烦,肿么办?难道一个个拷贝,那也太low啦,不是咱程序员该干的事儿,好的话不多说,开搞 ...
- 【Python】部署上手App后端服务器 - Linux环境搭建安装Python、Tornado、SQLAlchemy
基于阿里云服务器端环境搭建 文章目录 基于阿里云服务器端环境搭建 配置开发环境 安装 Python 3.8.2 安装 Tornado 安装 MySQL 安装 mysqlclient 安装 SQLAlc ...
- Linux下不同服务器间数据传输--转载
因为工作原因,需要经常在不同的服务器见进行文件传输,特别是大文件的传输,因此对linux下不同服务器间数据传输命令和工具进行了研究和总结.主要是rcp,scp,rsync,ftp,sftp,lftp, ...
- Linux下不同服务器间数据传输
因为工作原因,需要经常在不同的服务器见进行文件传输,特别是大文件的传输,因此对linux下不同服务器间数据传输命令和工具进行了研究和总结.主要是rcp,scp,rsync,ftp,sftp,lftp, ...
随机推荐
- NC的开发模型
2018-04-1622:35:12 NC的开发模型 系统前端为:客户UI代码,UI代码继承ToftPanel,UI代码通过调用远程组件和服务端进行交互,中间传递的数据模型为VO,远程组件调用底层的业 ...
- 实验吧_天下武功唯快不破&让我进去(哈希长度拓展攻击)
天下武功唯快不破 第一反应就去抓包,看到返回包的header中有FLAG的值,base64解码后得到下图所示 这就要求我们在请求头中post相应key的值,我直接在burp中尝试了多次都没有用,想起来 ...
- Event 发布与订阅(一)
前言 主要讲的是发布与订阅在Event中的一个简单实现用来加深理解. C #中的事件(Event)的理解: 事件具有以下属性:(From Events) 发行者确定何时引发事件:订户确定对事件作出何种 ...
- BZOJ3129: [Sdoi2013]方程
拓展Lucas+容斥原理 #include<cstdio> #include<cstdlib> #include<algorithm> #include<cs ...
- POJ 1486二分图的必要边
题意:有n个大小不等透明的幻灯片(只有轮廓和上面的数字可见)A.B.C.D.E…按顺序叠放在一起,现在知道每个幻灯片大小,由于幻灯片是透明的,所以能看到幻灯片上的数字(给出了每个数字的坐标,但不知道这 ...
- VK Cup 2017 - Round 2
FallDream打的AB都FFT了,只剩一个我打的C,没进前一百,之后看看马拉松复活赛有没机会呗. A. Voltage Keepsake 题目大意:n个东西,每个东西一开始有bi能源,每秒消耗ai ...
- bzoj 4010: [HNOI2015]菜肴制作
Description 知名美食家小 A被邀请至ATM 大酒店,为其品评菜肴. ATM 酒店为小 A 准备了 N 道菜肴,酒店按照为菜肴预估的质量从高到低给予 1到N的顺序编号,预估质量最高的菜肴编号 ...
- PHP查看本地文件夹及删除文件夹操作
查看文件夹(包括文件夹内所有的文件夹和文件) function descdir($dir){ if(is_dir($dir)){ if($dh=opendir($dir)){ while(($file ...
- async/await,了解一下?
上一篇博客我们在现实使用和面试角度讲解了Promise(原文可参考<面向面试题和实际使用谈promise>),但是Promise 的方式虽然解决了 callback hell,但是这种方式 ...
- bootstrap插件fileinput.js 出现出现$("#xxxx").fileinput({}); 不生效的情况解决
如果出现$("#xxxx").fileinput({}); 不生效的情况请将fileinput.js中最后几行注释掉: /* $(document).ready(function ...