我才刚刚开始学习RxJava,并且最近一直在为它而挣扎。我有一部分代码需要使用RxJava完成,并且我不希望在调用观察者的onSuccess()
或onError()
之前运行其余代码。我当前的实现如下:
@Override
public Bundle getAuthToken(AccountAuthenticatorResponse accountAuthenticatorResponse, Account account, String authTokenType, Bundle options) {
final AccountManager manager = AccountManager.get(mContext);
final String username = account.name;
String token = manager.peekAuthToken(account, authTokenType);
if (TextUtils.isEmpty(token)){
final String password = manager.getPassword(account);
if (password != null){
LoginClient client = ClientGenerator.createClient(LoginClient.class);
String encodedString = encodeClientIDAndSecret();
// Current implementation of observer
Single<TokenResponse> single = client.getRxAccessToken(encodedString, LoginClient.GRANT_TYPE, account.name, password, LoginClient.SCOPE);
single.map(new Function<TokenResponse, String>() {
@Override
public String apply(TokenResponse tokenResponse) throws Exception {
return tokenResponse.getAccessToken();
}
})
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new DisposableSingleObserver<String>() {
@Override
public void onSuccess(String accessToken) {
Log.d(TAG, "Here is the access token: " + accessToken);
}
@Override
public void onError(Throwable e) {
Log.d(TAG, "Unsuccessful response...");
}
});
}
}
token = manager.peekAuthToken(account, authTokenType);
if (!TextUtils.isEmpty(token)){
final Bundle result = new Bundle();
result.putString(AccountManager.KEY_ACCOUNT_NAME, account.name);
result.putString(AccountManager.KEY_ACCOUNT_TYPE, account.type);
result.putString(AccountManager.KEY_AUTHTOKEN, token);
return result;
} else {
final Intent intent = new Intent(mContext, LoginActivity.class);
intent.putExtra(AccountManager.KEY_ACCOUNT_AUTHENTICATOR_RESPONSE, accountAuthenticatorResponse);
intent.putExtra(LoginActivity.EXTRA_ACCOUNT_TYPE, account.type);
intent.putExtra(LoginActivity.EXTRA_AUTH_TOKEN_TYPE, authTokenType);
final Bundle bundle = new Bundle();
bundle.putParcelable(AccountManager.KEY_INTENT, intent);
return bundle;
}
}
有什么办法可以阻止我的其他代码在观察者完成其订阅之前运行?
为了提供一些更好的上下文,我正在尝试实现的是针对Android的自定义帐户身份验证器。用户首次登录时,他们将没有身份验证令牌,因此我使用RxJava获得了一个新令牌(如上所示)。然后,我需要将Bundle
返回到Android AccountManager
,这取决于我是否能够成功获取令牌。
我能够使用RxJava检索访问令牌。但是,如果我尝试在观察者外部读取令牌,则令牌为null,因为请求尚未完成...这就是为什么我希望以某种方式暂停其余代码的执行,直到订阅为止做完了。
一如既往,在此特定问题上的任何帮助将不胜感激:)
答案 0 :(得分:0)
如果这是你的单身
Single<TokenResponse> singleResponse = client.getRxAccessToken(
encodedString,
LoginClient.GRANT_TYPE,
account.name,
password,
LoginClient.SCOPE
);
然后尝试以下类似操作(getAuthToken()的返回类型需要更改):
@Override
public Single<Bundle> getAuthToken(AccountAuthenticatorResponse accountAuthenticatorResponse, Account account, String authTokenType, Bundle options) {
Return singleResponse
.flatMap(tokenResponse -> createBundle(tokenResponse))
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread());
}
现在创建这样的方法createBundle(TokenResponse tokenResponse)
private Single<Bundle> createBundle(TokenResponse tokenResponse) {
return Single.create(
e -> {
String token = tokenResponse.getAccessToken();
manager.peekAuthToken(account, authTokenType);
if (!TextUtils.isEmpty(token)){
final Bundle result = new Bundle();
result.putString(AccountManager.KEY_ACCOUNT_NAME, account.name);
result.putString(AccountManager.KEY_ACCOUNT_TYPE, account.type);
result.putString(AccountManager.KEY_AUTHTOKEN, token);
e.onSuccess(result);
} else {
final Intent intent = new Intent(mContext, LoginActivity.class);
intent.putExtra(
AccountManager.KEY_ACCOUNT_AUTHENTICATOR_RESPONSE, accountAuthenticatorResponse);
intent.putExtra(LoginActivity.EXTRA_ACCOUNT_TYPE, account.type);
intent.putExtra(LoginActivity.EXTRA_AUTH_TOKEN_TYPE, authTokenType);
final Bundle bundle = new Bundle();
bundle.putParcelable(AccountManager.KEY_INTENT, intent);
e.onSuccess(bundle);
}
}
);
}
现在,曾经在getAuthToken()
上进行订阅的人将获得Bundle
getAuthToken()
.subscribe(
bundle -> //Use bundle as needed,
error -> //Handle error scenario
)
希望这个答案有帮助。