#遇到問題: 當前開發的 App 遇到一個問題: 當請求某個介面時,由於 token 已經失效,所以介面會報錯。但是產品經理希望 app 能夠馬上刷新 token ,然後重覆請求剛纔那個介面,這個過程對用戶來說是無感的。>也就是靜默自動登錄,然後繼續請求:>請求 A 介面-》伺服器返回 token ...
#遇到問題:
當前開發的 App 遇到一個問題:
當請求某個介面時,由於 token 已經失效,所以介面會報錯。
但是產品經理希望 app 能夠馬上刷新 token ,然後重覆請求剛纔那個介面,這個過程對用戶來說是無感的。
>
也就是靜默自動登錄,然後繼續請求:
>
請求 A 介面-》伺服器返回 token 過期-》請求 token 刷新介面-》請求 A 介面
>
要實現上述需求的話,大家會如何實現呢?
#解決方案:
思路:
1.通過攔截器,獲取返回的數據
2.判斷token是否過期
3.如果token過期則刷新token
4.使用最新的token,重新請求網路數據
1 /** 2 * 全局自動刷新Token的攔截器 3 */ 4 public class TokenInterceptor implements Interceptor { 5 6 @Override 7 public Response intercept(Chain chain) throws IOException { 8 Request request = chain.request(); 9 Response response = chain.proceed(request); 10 LogUtil.print("response.code=" + response.code()); 11 12 if (isTokenExpired(response)) {//根據和服務端的約定判斷token過期 13 LogUtil.print("靜默自動刷新Token,然後重新請求數據"); 14 //同步請求方式,獲取最新的Token 15 String newSession = getNewToken(); 16 //使用新的Token,創建新的請求 17 Request newRequest = chain.request() 18 .newBuilder() 19 .header("Cookie", "JSESSIONID=" + newSession) 20 .build(); 21 //重新請求 22 return chain.proceed(newRequest); 23 } 24 return response; 25 } 26 27 /** 28 * 根據Response,判斷Token是否失效 29 * 30 * @param response 31 * @return 32 */ 33 private boolean isTokenExpired(Response response) { 34 if (response.code() == 404) { 35 return true; 36 } 37 return false; 38 } 39 40 /** 41 * 同步請求方式,獲取最新的Token 42 * 43 * @return 44 */ 45 private String getNewToken() throws IOException { 46 // 通過一個特定的介面獲取新的token,此處要用到同步的retrofit請求 47 Response_Login loginInfo = CacheManager.restoreLoginInfo(BaseApplication.getContext()); 48 String username = loginInfo.getUserName(); 49 String password = loginInfo.getPassword(); 50 51 LogUtil.print("loginInfo=" + loginInfo.toString()); 52 Call<Response_Login> call = WebHelper.getSyncInterface().synclogin(new Request_Login(username, password)); 53 loginInfo = call.execute().body(); 54 LogUtil.print("loginInfo=" + loginInfo.toString()); 55 56 loginInfo.setPassword(password); 57 CacheManager.saveLoginInfo(loginInfo); 58 return loginInfo.getSession(); 59 } 60 }
然後配置下OkHttp
HttpLoggingInterceptor logging = new HttpLoggingInterceptor(); logging.setLevel(HttpLoggingInterceptor.Level.BODY); OkHttpClient client = new OkHttpClient.Builder() .connectTimeout(15, TimeUnit.SECONDS) .readTimeout(300, TimeUnit.SECONDS) .writeTimeout(300, TimeUnit.SECONDS) .cache(new Cache(FileConstants.HTTP_CACHE_DIR, FileConstants.CACHE_SIZE)) .addInterceptor(interceptor) // .addInterceptor(new MockInterceptor()) .addInterceptor(new TokenInterceptor()) // .addInterceptor(new RetryIntercepter(3)) .addInterceptor(logging) .build();