私は自分のアプリケーションの1つにokhttpとretrofitを使用していますが、RxJavaを使用してさまざまなスレッドで作業する多くのリクエストがあります。リクエストのいずれかでSocketExceptionが発生し、その後、サーバーでリクエストが到達できないことがあります。
例: 1つのサンプルを共有できます。
共有できるコードからの変更されたスニペットはほとんどありません。
Subscription details = api.details("keyword")
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Subscriber<Model>() {
@Override
public void onCompleted() {
Timber.d("onCompleted(): ");
}
@Override
public void onError(Throwable e) {
Timber.e(e, "onError(): ");
}
@Override
public void onNext(final Model details) {
Timber.d("onNext(): ");
}
});
3つのコールバックすべてから何も呼び出されませんでした。
Stackstace:
11-17 15:50:53.991 16595-10314/ D/OkHttp: --> GET API_REQUEST_URL
11-17 15:50:53.991 16595-10314/ D/OkHttp: Host: Host_NAME
11-17 15:50:53.991 16595-10314/ D/OkHttp: Connection: Keep-Alive
11-17 15:50:53.991 16595-10314/ D/OkHttp: Accept-Encoding: gzip
11-17 15:50:53.991 16595-10314/ D/OkHttp: User-Agent: okhttp/3.4.2
11-17 15:50:53.991 16595-10314/ D/OkHttp: User-Agent: Android-App
11-17 15:50:53.991 16595-10314/ D/OkHttp: --> END GET
11-17 15:50:53.992 16595-10314/ D/OkHttp: <-- HTTP FAILED: Java.net.SocketException: Socket closed
その他のIOException
11-17 16:36:04.274 28523-29137/ D/OkHttp: Host: Host_NAME
11-17 16:36:04.274 28523-29137/ D/OkHttp: Connection: Keep-Alive
11-17 16:36:04.274 28523-29137/ D/OkHttp: Accept-Encoding: gzip
11-17 16:36:04.274 28523-29137/ D/OkHttp: User-Agent: okhttp/3.4.2
11-17 16:36:04.274 28523-29137/ D/OkHttp: User-Agent: Android-App
11-17 16:36:04.274 28523-29137/ D/OkHttp: --> END GET
11-17 16:36:04.282 28523-29137/ D/OkHttp: <-- HTTP FAILED: Java.io.IOException: unexpected end of stream on okhttp3.Address@6924d4a0
サブスクリプション私は頻繁に正常に動作していると呼ぼうとしていますが、上記のエラーが発生して動作を停止することがあります。リクエストがサーバーに到達できないAferwords。
改造インターフェースからの方法
@GET("/v1/app_endpoint/{keyword}/")
Observable<Model> details(@Path("keyword") String keyword);
必要に応じて構成を確認するためのいくつかのヘルパーメソッドを以下に示します。
public static Retrofit retrofit() {
return provideRetrofit(provideGson(), provideOkHttpClient());
}
public static Gson provideGson() {
GsonBuilder gsonBuilder = new GsonBuilder();
gsonBuilder.addSerializationExclusionStrategy(new AnnotationExclusionStrategy());
gsonBuilder.addDeserializationExclusionStrategy(new AnnotationExclusionStrategy());
gsonBuilder.setExclusionStrategies(new ExclusionStrategy() {
@Override
public boolean shouldSkipField(FieldAttributes f) {
return f.getDeclaringClass().equals(RealmObject.class);
}
@Override
public boolean shouldSkipClass(Class<?> clazz) {
return false;
}
});
gsonBuilder.registerTypeAdapterFactory(AutoValueGsonTypeAdapterFactory.create());
return gsonBuilder.create();
}
public static OkHttpClient provideOkHttpClient() {
return new OkHttpClient.Builder()
.connectTimeout(30, TimeUnit.SECONDS)
.writeTimeout(600, TimeUnit.SECONDS)
.readTimeout(1, TimeUnit.MINUTES)
.retryOnConnectionFailure(true)
.addNetworkInterceptor(new Interceptor() {
@Override
public okhttp3.Response intercept(Chain chain) throws IOException {
Request newRequest = chain.request().newBuilder()
.addHeader("Content-Type", "application/json")
.addHeader("User-Agent", "Android-App")
.build();
return chain.proceed(newRequest);
}
})
.addNetworkInterceptor(new HttpLoggingInterceptor().setLevel(HttpLoggingInterceptor.Level.BODY))
.build();
}
public static Retrofit provideRetrofit(Gson gson, OkHttpClient okHttpClient) {
return new Retrofit.Builder()
.baseUrl(BuildConfig.API_URL)
.client(okHttpClient)
.addConverterFactory(GsonConverterFactory.create(gson))
.addCallAdapterFactory(RxJavaCallAdapterFactory.createWithScheduler(Schedulers.io()))
.build();
}
依存関係
// Networking
compile 'com.squareup.okhttp3:okhttp:3.4.2'
compile 'com.squareup.retrofit2:retrofit:2.1.0'
compile 'com.squareup.retrofit2:converter-gson:2.1.0'
compile 'com.squareup.retrofit2:adapter-rxjava:2.1.0'
compile 'com.squareup.okhttp3:logging-interceptor:3.4.1'
私の場合、私が得ていたのは、突然の予期しないunsubscribe()
呼び出しであり、_HTTP FAILED: Java.net.SocketException: Socket closed
_エラーが発生し、onCompleted()
onError()
またはonNext()
コールバック-ディスカッションをチェックして答えてください ここ 。
したがって、そのケースを処理できるようにするには、次を追加するだけです。
_.doOnUnsubscribe(() -> { /*handle logic here*/ })
_
Observable
に移動し、そこで状態をリセットしてみてください。これにより、通常のアクティビティを再開できるようになります。
私の場合、間違ったURLをリクエストしてエラーが発生しましたERR_CONNECTION_REFUSED
。