当服务器连接失败时,Retrofit + Rxjava需要很长时间才能调用onError

Ali*_*ali 5 networking android connectivity rx-java retrofit

我正在使用Retrofit + Rxjava从服务器获取实体列表,当我的设计任务失败时,首先检查Internet连接,然后检查与服务器的连接doOnError方法Observable.

当客户端未连接到Internet时,doOnError会在合理的时间内调用并且用户会收到错误消息,但问题是当Internet连接时我得到错误的端口或域(检查服务器问题错误)需要很长时间(约1分钟或更长时间)这真的很烦人.我怎样才能减少这个时间,原因是什么?

我如何检查Internet和服务器连接:

public static boolean checkConnection(String ipOrUrl, int port) {
    try {
        int timeoutMs = 100;
        Socket socket = new Socket();
        SocketAddress soketAddress = new InetSocketAddress(ipOrUrl, port);
        socket.connect(soketAddress, timeoutMs);
        socket.close();
        return true;
    } catch (IOException e) {
        e.printStackTrace();
        return false;
    }
}
Run Code Online (Sandbox Code Playgroud)

我如何尝试获取实体列表:

    foodgetRetorfitService.getRestaurants()
            .subscribeOn(Schedulers.newThread())
            .observeOn(AndroidSchedulers.mainThread())
//here I'm checking the internet and server connection in the onlineTask if retrofit fail
//and if clients get Online this ( getRestaurantFromServer ) is called again.
            .doOnError(error -> {
                error.printStackTrace();
                NetworkUtils.doOnlineTask(new OnlineTask() {
                    public void doWhenOnline() {
                        getResturantsFromServer();
                    }
                }, true);
            })
            .subscribe(new Observer<List<Restaurant>>() {
                @Override
                public void onNext(List<Restaurant> restaurants) {
                    restaurantItemAdapter.updateAdapterData(restaurants);
                }

                @Override
                public void onError(Throwable e) {
                    Log.e(TAG, "onError: rxjava and retrofit error : can't get restaurant list");
                    e.printStackTrace();
                }
            });
Run Code Online (Sandbox Code Playgroud)

如何doOnlineTask实施

    public static void doOnlineTask(OnlineTask onlineTask, boolean autoRetry, int retryTimeout) {
        NetworkUtils.isOnline(autoRetry)
                .subscribeOn(Schedulers.newThread())
                .observeOn(AndroidSchedulers.mainThread())
                .doOnError(error -> {
//here I check the Exception type ( I raised them when checking the connection )
                    if (error instanceof InternetConnectionException)
                        onlineTask.doWhenInternetFaild();

                    if (error instanceof ServerConnectionException)
                        onlineTask.doWhenServerFaild();
                })
                .retryWhen(t -> t.delay(2, TimeUnit.SECONDS))
                .subscribe(result -> {
                            if (result.equals(NetworkStatus.CONNECTED))
                                onlineTask.doWhenOnline();
                            else {
                                if (result.equals(NetworkStatus.INTERNET_FAILD))
                                    onlineTask.doWhenInternetFaild();
                                else if (result.equals(NetworkStatus.SERVER_FAILD))
                                    onlineTask.doWhenInternetFaild();
                            }
                        }, error -> error.printStackTrace()
                );
    }
Run Code Online (Sandbox Code Playgroud)

onlineTask 只是一个抽象的类

abstract public void doWhenOnline();

public void doWhenInternetFaild() {
   //implemented somehow
}


public void doWhenServerFaild() {
   //implemented somehow
}
Run Code Online (Sandbox Code Playgroud)

我尝试了什么:

我猜测它是超时问题所以我改变了Retrofit超时,OkHttpClient它没有用.我也改变了自己设定的超时时间,并减少了它们.不工作

And*_*kin 0

我假设您正在使用 OkHttp 客户端来配合 Retrofit。您正在寻找的很可能是连接超时。

您可以在构建客户端时设置不同的超时。

OkHttpClient client = new OkHttpClient().newBuilder()
                .connectTimeout(30, TimeUnit.SECONDS)
                .readTimeout(60, TimeUnit.SECONDS)
                .writeTimeout(15, TimeUnit.SECONDS)
                .build();
Run Code Online (Sandbox Code Playgroud)