前言:
前面的學習基本上已經可以完成開發需求了,但是在專案中有時會遇到對請求做個快取,當沒網路的時候優先載入本地快取,基於這個需求我們來學習一直okHttp的Cache-Control。
okHttp相關文章地址:
- Android okHttp網路請求之Get/Post請求
- Android okHttp網路請求之檔案上傳下載
- Android okHttp網路請求之Json解析
- Android okHttp網路請求之快取控制Cache-Control
- Android okHttp網路請求之Retrofit+Okhttp組合
Cache-Control:
Cache-Control指定請求和響應遵循的快取機制。在請求訊息或響應訊息中設定Cache-Control並不會修改另一個訊息處理過程中的快取處理過程。請求時的快取指令有下幾種:
- Public指示響應可被任何快取區快取。
- Private指示對於單個使用者的整個或部分響應訊息,不能被共享快取處理。這允許伺服器僅僅描述當使用者的部分響應訊息,此響應訊息對於其他使用者的請求無效。
- no-cache指示請求或響應訊息不能快取
- no-store用於防止重要的資訊被無意的釋出。在請求訊息中傳送將使得請求和響應訊息都不使用快取。
- max-age指示客戶機可以接收生存期不大於指定時間(以秒為單位)的響應。
- min-fresh指示客戶機可以接收響應時間小於當前時間加上指定時間的響應。
- max-stale指示客戶機可以接收超出超時期間的響應訊息。如果指定max-stale訊息的值,那麼客戶機可以接收超出超時期指定值之內的響應訊息。
CacheControl.java類介紹:
1.)常用的函式:如下程式碼,裡面已經加了註釋就不一一解釋了,每個函式都是對應一個快取指令設定
final CacheControl.Builder builder = new CacheControl.Builder(); builder.noCache();//不使用快取,全部走網路 builder.noStore();//不使用快取,也不儲存快取 builder.onlyIfCached();//只使用快取 builder.noTransform();//禁止轉碼 builder.maxAge(10, TimeUnit.MILLISECONDS);//指示客戶機可以接收生存期不大於指定時間的響應。 builder.maxStale(10, TimeUnit.SECONDS);//指示客戶機可以接收超出超時期間的響應訊息 builder.minFresh(10, TimeUnit.SECONDS);//指示客戶機可以接收響應時間小於當前時間加上指定時間的響應。 CacheControl cache = builder.build();//cacheControl
2.)兩個CacheControl常量介紹:
CacheControl.FORCE_CACHE; //僅僅使用快取 CacheControl.FORCE_NETWORK;// 僅僅使用網路
舉例,我們設定一個有效期為10秒的CacheControl
final CacheControl.Builder builder = new CacheControl.Builder(); builder.maxAge(10, TimeUnit.MILLISECONDS); CacheControl cache = builder.build();
3.)請求時如何使用
final CacheControl.Builder builder = new CacheControl.Builder(); builder.maxAge(10, TimeUnit.MILLISECONDS); CacheControl cache = builder.build(); final Request request = new Request.Builder().cacheControl(cache).url(requestUrl).build(); final Call call = mOkHttpClient.newCall(request);// call.enqueue(new Callback() { @Override public void onFailure(Call call, IOException e) { failedCallBack("訪問失敗", callBack); Log.e(TAG, e.toString()); } @Override public void onResponse(Call call, Response response) throws IOException { if (response.isSuccessful()) { String string = response.body().string(); Log.e(TAG, "response ----->" + string); successCallBack((T) string, callBack); } else { failedCallBack("伺服器錯誤", callBack); } } }); return call; } catch (Exception e) { Log.e(TAG, e.toString()); }
以上如果cache沒有過去會直接返回cache而不會發起網路請求,若過期會自動發起網路請求。注意:如果您使用FORCE_CACHE和網路的響應需求,OkHttp則會返回一個504提示,告訴你不可滿足請求響應。所以我們加一個判斷在沒有網路的情況下使用
//判斷網路是否連線 boolean connected = NetworkUtil.isConnected(context); if (!connected) { request = request.newBuilder().cacheControl(CacheControl.FORCE_CACHE).build(); }
okHtitp知識擴充套件:
1.)Interceptor攔截器,見名知意就是攔截操作,這裡用來攔截Request對其做一些特殊處理,舉例:比如上面我們使用到了CacheControl,我們怎麼攔截一個請求在網路不可用的時候使用CacheControl.FORCE_CACHE;
OkHttpClient.Builder newBuilder = mOkHttpClient.newBuilder(); newBuilder.addInterceptor(new Interceptor() { @Override public Response intercept(Chain chain) throws IOException { Request request = chain.request(); boolean connected = NetworkUtil.isConnected(context); if (!connected) { request = request.newBuilder().cacheControl(CacheControl.FORCE_CACHE).build(); } Response response = chain.proceed(request); return response; } });
2.)OkHttp 提供了對使用者認證的支援。當 HTTP 響應的狀態程式碼是 401 時,OkHttp 會從設定的 Authenticator 物件中獲取到新的 Request 物件並再次嘗試發出請求。Authenticator 介面中的 authenticate 方法用來提供進行認證的 Request 物件.
OkHttpClient client = new OkHttpClient(); client.newBuilder().authenticator(new Authenticator() { @Override public Request authenticate(Route route, Response response) throws IOException { String credential = Credentials.basic("user", "password"); return response.request().newBuilder() .header("Authorization", credential) .build(); } });
小結:okHttp的簡單使用到此介紹完畢,至於很多高階使用還有待研究。接下來準備研究下OkHttp與retrofit結合使用。