OK를 사용하여 리트로피팅 가능 오프라인 시 캐시 데이터 사용
HTTP 응답을 캐시하기 위해 Retrofit & OKHttp를 사용하려고 합니다.나는 이 요지를 따랐고, 결국 이 코드로 끝났다.
File httpCacheDirectory = new File(context.getCacheDir(), "responses");
HttpResponseCache httpResponseCache = null;
try {
httpResponseCache = new HttpResponseCache(httpCacheDirectory, 10 * 1024 * 1024);
} catch (IOException e) {
Log.e("Retrofit", "Could not create http cache", e);
}
OkHttpClient okHttpClient = new OkHttpClient();
okHttpClient.setResponseCache(httpResponseCache);
api = new RestAdapter.Builder()
.setEndpoint(API_URL)
.setLogLevel(RestAdapter.LogLevel.FULL)
.setClient(new OkClient(okHttpClient))
.build()
.create(MyApi.class);
캐시 제어 헤더가 있는 MyApi입니다.
public interface MyApi {
@Headers("Cache-Control: public, max-age=640000, s-maxage=640000 , max-stale=2419200")
@GET("/api/v1/person/1/")
void requestPerson(
Callback<Person> callback
);
먼저 온라인으로 요청하고 캐시 파일을 확인합니다.JSON을 사용하다 오프라인으로 '오프라인이요'라는 가 뜬다.RetrofitError UnknownHostException
복고풍?
편집: OKHttp 2.0.x 이후HttpResponseCache
Cache
,setResponseCache
setCache
Retrofit 2.x 편집:
OkHttp 대행 수신기는 오프라인일 때 캐시에 액세스하는 올바른 방법입니다.
1) 가로채기 작성:
private static final Interceptor REWRITE_CACHE_CONTROL_INTERCEPTOR = new Interceptor() {
@Override public Response intercept(Chain chain) throws IOException {
Response originalResponse = chain.proceed(chain.request());
if (Utils.isNetworkAvailable(context)) {
int maxAge = 60; // read from cache for 1 minute
return originalResponse.newBuilder()
.header("Cache-Control", "public, max-age=" + maxAge)
.build();
} else {
int maxStale = 60 * 60 * 24 * 28; // tolerate 4-weeks stale
return originalResponse.newBuilder()
.header("Cache-Control", "public, only-if-cached, max-stale=" + maxStale)
.build();
}
}
2) 셋업 클라이언트:
OkHttpClient client = new OkHttpClient();
client.networkInterceptors().add(REWRITE_CACHE_CONTROL_INTERCEPTOR);
//setup cache
File httpCacheDirectory = new File(context.getCacheDir(), "responses");
int cacheSize = 10 * 1024 * 1024; // 10 MiB
Cache cache = new Cache(httpCacheDirectory, cacheSize);
//add cache to the client
client.setCache(cache);
3) 개조에 클라이언트 추가
Retrofit retrofit = new Retrofit.Builder()
.baseUrl(BASE_URL)
.client(client)
.addConverterFactory(GsonConverterFactory.create())
.build();
@kosiara - Bartosz Kosarzyki의 답변도 확인하십시오.응답에서 일부 헤더를 제거해야 할 수 있습니다.
OKHttp 2.0.x (원래 응답을 확인합니다)
2.0 OKHttp 2.0.x 이후HttpResponseCache
Cache
,setResponseCache
setCache
setCache
음음음같 뭇매하다
File httpCacheDirectory = new File(context.getCacheDir(), "responses");
Cache cache = null;
try {
cache = new Cache(httpCacheDirectory, 10 * 1024 * 1024);
} catch (IOException e) {
Log.e("OKHttp", "Could not create http cache", e);
}
OkHttpClient okHttpClient = new OkHttpClient();
if (cache != null) {
okHttpClient.setCache(cache);
}
String hostURL = context.getString(R.string.host_url);
api = new RestAdapter.Builder()
.setEndpoint(hostURL)
.setClient(new OkClient(okHttpClient))
.setRequestInterceptor(/*rest of the answer here */)
.build()
.create(MyApi.class);
원답:
'알다'가 있어야 합니다.Cache-Control: public
만들다OkClient
이치노
사용 가능한 경우 을 추가해야 .Cache-Control: max-age=0
request.이 답변은 매개 변수를 사용하여 수행하는 방법을 보여 줍니다.사용 방법은 다음과 같습니다.
RestAdapter.Builder builder= new RestAdapter.Builder()
.setRequestInterceptor(new RequestInterceptor() {
@Override
public void intercept(RequestFacade request) {
request.addHeader("Accept", "application/json;versions=1");
if (MyApplicationUtils.isNetworkAvailable(context)) {
int maxAge = 60; // read from cache for 1 minute
request.addHeader("Cache-Control", "public, max-age=" + maxAge);
} else {
int maxStale = 60 * 60 * 24 * 28; // tolerate 4-weeks stale
request.addHeader("Cache-Control",
"public, only-if-cached, max-stale=" + maxStale);
}
}
});
위의 모든 Anwers가 나에게 작동하지는 않았다.retrofit 2.0.0-beta2에서 오프라인 캐시를 구현하려고 했습니다.요격기를 추가했습니다.okHttpClient.networkInterceptors()
했지만 수신했습니다.java.net.UnknownHostException
캐시를 오프라인으로 사용하려고 했을 때. 보니 나는 ★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★okHttpClient.interceptors()
뿐만 아니라.
가 플래시 스토리지를 반환했기 플래시 입니다.Pragma:no-cache
OkHttp로 하다요청 헤더 값을 수정한 후에도 오프라인 캐시가 작동하지 않았습니다. 시행착오를 겪은 하지 않고 .response.newBuilder().removeHeader("Pragma");
개조: 2.0.0-베타2, OkHttp: 2.5.0
OkHttpClient okHttpClient = createCachedClient(context);
Retrofit retrofit = new Retrofit.Builder()
.client(okHttpClient)
.baseUrl(API_URL)
.addConverterFactory(GsonConverterFactory.create())
.build();
service = retrofit.create(RestDataResource.class);
...
private OkHttpClient createCachedClient(final Context context) {
File httpCacheDirectory = new File(context.getCacheDir(), "cache_file");
Cache cache = new Cache(httpCacheDirectory, 20 * 1024 * 1024);
OkHttpClient okHttpClient = new OkHttpClient();
okHttpClient.setCache(cache);
okHttpClient.interceptors().add(
new Interceptor() {
@Override
public Response intercept(Chain chain) throws IOException {
Request originalRequest = chain.request();
String cacheHeaderValue = isOnline(context)
? "public, max-age=2419200"
: "public, only-if-cached, max-stale=2419200" ;
Request request = originalRequest.newBuilder().build();
Response response = chain.proceed(request);
return response.newBuilder()
.removeHeader("Pragma")
.removeHeader("Cache-Control")
.header("Cache-Control", cacheHeaderValue)
.build();
}
}
);
okHttpClient.networkInterceptors().add(
new Interceptor() {
@Override
public Response intercept(Chain chain) throws IOException {
Request originalRequest = chain.request();
String cacheHeaderValue = isOnline(context)
? "public, max-age=2419200"
: "public, only-if-cached, max-stale=2419200" ;
Request request = originalRequest.newBuilder().build();
Response response = chain.proceed(request);
return response.newBuilder()
.removeHeader("Pragma")
.removeHeader("Cache-Control")
.header("Cache-Control", cacheHeaderValue)
.build();
}
}
);
return okHttpClient;
}
...
public interface RestDataResource {
@GET("rest-data")
Call<List<RestItem>> getRestData();
}
솔루션:
private BackendService() {
httpCacheDirectory = new File(context.getCacheDir(), "responses");
int cacheSize = 10 * 1024 * 1024; // 10 MiB
Cache cache = new Cache(httpCacheDirectory, cacheSize);
httpClient = new OkHttpClient.Builder()
.addNetworkInterceptor(REWRITE_RESPONSE_INTERCEPTOR)
.addInterceptor(OFFLINE_INTERCEPTOR)
.cache(cache)
.build();
Retrofit retrofit = new Retrofit.Builder()
.baseUrl("https://api.backend.com")
.client(httpClient)
.addConverterFactory(GsonConverterFactory.create())
.build();
backendApi = retrofit.create(BackendApi.class);
}
private static final Interceptor REWRITE_RESPONSE_INTERCEPTOR = chain -> {
Response originalResponse = chain.proceed(chain.request());
String cacheControl = originalResponse.header("Cache-Control");
if (cacheControl == null || cacheControl.contains("no-store") || cacheControl.contains("no-cache") ||
cacheControl.contains("must-revalidate") || cacheControl.contains("max-age=0")) {
return originalResponse.newBuilder()
.header("Cache-Control", "public, max-age=" + 10)
.build();
} else {
return originalResponse;
}
};
private static final Interceptor OFFLINE_INTERCEPTOR = chain -> {
Request request = chain.request();
if (!isOnline()) {
Log.d(TAG, "rewriting request");
int maxStale = 60 * 60 * 24 * 28; // tolerate 4-weeks stale
request = request.newBuilder()
.header("Cache-Control", "public, only-if-cached, max-stale=" + maxStale)
.build();
}
return chain.proceed(request);
};
public static boolean isOnline() {
ConnectivityManager cm = (ConnectivityManager) MyApplication.getApplication().getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo netInfo = cm.getActiveNetworkInfo();
return netInfo != null && netInfo.isConnectedOrConnecting();
}
답은 '예'입니다. 위의 답변을 바탕으로 가능한 모든 사용 사례를 확인하기 위해 유닛 테스트를 작성하기 시작했습니다.
- 오프라인일 때 캐시 사용
- 먼저 캐시된 응답을 만료될 때까지 사용한 후 네트워크
- 일부 요청에 대해 먼저 네트워크를 사용한 후 캐시 사용
- 일부 응답에 대해 캐시에 저장 안 함
OKHttp 캐시를 쉽게 설정하기 위해 작은 도우미 lib를 구축했습니다.관련 unittest는 여기 Github에서 보실 수 있습니다.https://github.com/ncornette/OkCacheControl/blob/master/okcache-control/src/test/java/com/ncornette/cache/OkCacheControlTest.java
오프라인에서의 캐시 사용을 나타내는 Unittest:
@Test
public void test_USE_CACHE_WHEN_OFFLINE() throws Exception {
//given
givenResponseInCache("Expired Response in cache", -5, MINUTES);
given(networkMonitor.isOnline()).willReturn(false);
//when
//This response is only used to not block when test fails
mockWebServer.enqueue(new MockResponse().setResponseCode(404));
Response response = getResponse();
//then
then(response.body().string()).isEqualTo("Expired Response in cache");
then(cache.hitCount()).isEqualTo(1);
}
보시다시피 캐시는 유효기간이 지났어도 사용할 수 있습니다.도움이 되길 바랍니다.
@kosiara-bartosz-kasarzycki의 답변을 바탕으로 retrofit, okhttp, rxjava 및 guava를 사용하여 메모리로부터 올바르게 로드하는 샘플 프로젝트를 작성했습니다.https://github.com/digitalbuddha/StoreDemo
Retrofit2 및 OkHTTP3를 사용한 캐시:
OkHttpClient client = new OkHttpClient
.Builder()
.cache(new Cache(App.sApp.getCacheDir(), 10 * 1024 * 1024)) // 10 MB
.addInterceptor(new Interceptor() {
@Override public Response intercept(Chain chain) throws IOException {
Request request = chain.request();
if (NetworkUtils.isNetworkAvailable()) {
request = request.newBuilder().header("Cache-Control", "public, max-age=" + 60).build();
} else {
request = request.newBuilder().header("Cache-Control", "public, only-if-cached, max-stale=" + 60 * 60 * 24 * 7).build();
}
return chain.proceed(request);
}
})
.build();
NetworkUtils.isNetworkAvailable() 정적 메서드:
public static boolean isNetworkAvailable(Context context) {
ConnectivityManager cm =
(ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo activeNetwork = cm.getActiveNetworkInfo();
return activeNetwork != null &&
activeNetwork.isConnectedOrConnecting();
}
그런 다음 클라이언트를 개조 빌더에 추가합니다.
Retrofit retrofit = new Retrofit.Builder()
.baseUrl(BASE_URL)
.client(client)
.addConverterFactory(GsonConverterFactory.create())
.build();
출처 : https://newfivefour.com/android-retrofit2-okhttp3-cache-network-request-offline.html
차려!캐시 내 OkHttp 빌드는 다음 시간에만 작동합니다.GET
method (위 솔루션 참조)캐시하고 싶은 경우POST
요청, 직접 구현해야 합니다.
언급URL : https://stackoverflow.com/questions/23429046/can-retrofit-with-okhttp-use-cache-data-when-offline
'IT이야기' 카테고리의 다른 글
v-bind를 사용하여 Vue가 거짓 특성을 표시하도록 강제하려면 어떻게 합니까? (0) | 2022.07.04 |
---|---|
Vuex 하위 구성 요소는 이 항목에 액세스할 수 없습니다.$store(표준) (0) | 2022.07.04 |
Vue2의 조건부 소품 (0) | 2022.07.04 |
하나의 플랫폼에서 모든 데이터 유형의 모든 데이터 포인터가 동일한 크기입니까? (0) | 2022.07.04 |
"소프트웨어로 인해 연결이 중단되었습니다: 소켓 쓰기 오류"의 공식 이유 (0) | 2022.07.04 |