Android/RxJava 如何链式处理网络请求并在失败时重试

5
我正在尝试在Android中使用RxJava链接网络请求,如果失败则进行重试。我一直在查找StackOverflow上如何在不陷入原生Android和Retrofit混乱的回调地狱的情况下完成此操作。我可以在异步任务中通过以下步骤(下面是我需要执行的步骤)完成登录。
  1. 使用用户名/密码登录用户
  2. 如果响应成功,则请求SSO令牌
  3. 使用该SSO令牌调用启动服务
我曾经在AsyncTask中这样做,但只有在登录时才有效。然而,当应用程序在使用过程中SSO令牌过期时,我需要每次看到它过期时都进行请求。
AsyncTask的代码如下:
private class LoginUserTask extends AsyncTask<Void, Void, Void> {
    private final String LOG_TAG = LoginUserTask.class.getSimpleName();

    @Override
    protected Void doInBackground(Void... params) {
        OkHttpClient httpClient = ((TribeSocial) getApplication()).getHttpClient();

        // Login User
        GroupDockService groupDockLoginService =
                GroupDockServiceGenerator
                        .createService(GroupDockService.class,
                                httpClient, GroupDockUser.class, new GroupDockUserDeserializer());
        GroupDockUser groupDockUser = groupDockLoginService.loginUser("Tribe", username, password);
        Utility.saveAccountSubdomain(mContext, groupDockUser.getGroupDockSubdomain().getSubdomain());

        // Get Sso Token
        GroupDockService groupDockService = GroupDockServiceGenerator
                .createService(GroupDockService.class, httpClient);
        GroupDockSsoResponse ssoResponse =
                groupDockService.getSsoToken(Utility.getAccountSubdomain(mContext), true);
        Utility.saveSsoToken(mContext, ssoResponse.getSsoToken());

        // Sign in user into Tribe service
        TribeSocialService tribeSocialLaunchService =
                TribeServiceGenerator.createService(TribeSocialService.class,
                        httpClient, new LenientGsonConverter(new Gson()));
        tribeSocialLaunchService.launch(Utility.getSsoToken(mContext));

        // Get User id and save it to SharedPreferences
        TribeSocialService tribeSocialWhoAmIService =
                TribeServiceGenerator.createService(TribeSocialService.class, httpClient,
                        User.class, new WhoAmIDeserializer());
        User tribeUser = tribeSocialWhoAmIService.whoami();
        Utility.saveUserId(mContext, tribeUser.getId());

        return null;
    }

    @Override
    protected void onPostExecute(Void v) {
        Utility.launchMainActivity(mContext);
    }
}

我尝试使用RxJava实现这个功能的方式是:

GroupDockService groupDockLoginService =
        GroupDockServiceGenerator
                .createService(GroupDockService.class,
                        mHttpClient, GroupDockUser.class, new GroupDockUserDeserializer());

groupDockLoginService
        .loginUserRx("Tribe", username, password)
        .subscribeOn(Schedulers.newThread())
        .observeOn(AndroidSchedulers.mainThread())
        .subscribe(new Subscriber<GroupDockUser>() {
            @Override
            public void onCompleted() {

            }

            @Override
            public void onError(Throwable e) {
                Log.e(LOG_TAG, "I have errored out: " + e.getMessage());
                Toast.makeText(mContext,
                        getString(R.string.username_and_or_password_is_incorrect),
                        Toast.LENGTH_SHORT).show();
            }

            @Override
            public void onNext(GroupDockUser groupDockUser) {
                Utility.saveUsernamePassword(mContext, username, password);
                new LoginUserTask().execute();
            }
        });

在这段代码中,我需要执行以下操作:
  1. 当loginUserRx成功时,我需要将用户名和密码保存到SavedPreferences中
  2. 然后我需要发起网络请求以获取sso令牌
  3. 之后,我需要进行第三个和最后一个网络请求来启动。
目前的代码只是调用登录用户的函数,并在成功后启动异步任务。理想情况下,我希望能够将所有请求链接在一起。我已经尝试使用flatMap、map等方法,但无法弄清楚如何按照上述步骤链接这些调用。
有人可以给我指点一下吗?谢谢。

retry 怎么样?我不认为 map 或 flatmap 能帮助你实现想要的功能。 - OneCricketeer
1个回答

10

看一下这个例子。希望它能激发你。

service
        //do login
        .loginUser()
        //side effect: save into Shared Preference
        .doOnNext( )
        //get SSO
        .flatMap()
        //do any other side effects
        .doOnNext()
        //do final network request
        .flatMap()
        //in case of error retry after 5 seconds
        .retryWhen(observable -> Observable.timer(5, TimeUnit.SECONDS))
        .subscribeOn(Schedulers.computation())
        .observeOn(AndroidSchedulers.mainThread())
        //update you UI
        .subscribe();

关于重试的更多阅读资料


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接