我正在尝试将 Firestore方法包装到CompletableEmitter中,以便能够在成功/完成时返回Completable.complete(),而在失败时返回Completable.error()。 (任务是Firestore操作,用于更新集合中的文档(该文档中的1个字段))
This是我要包装成Completable的firestore method
,它返回一个Task<Void>
。这是没有包装在Completable中的代码:
public static Completable updateDocument(DocumentReference ref, Map<String,Object> fieldsAndValues){
ref.update(fieldsAndValues)
.addOnSuccessListener(aVoid -> {
Timber.d("SUCCESS");
}).addOnCompleteListener(task -> {
Timber.d("COMPLETE");
}).addOnFailureListener(e -> {
Timber.d("ERROR");
});
return Completable.complete(); //just for the sake of returning complete to finish the method...
}
上面的代码按原样执行。
这是我包装该代码的尝试:
public static Completable updateDocument(DocumentReference ref, Map<String,Object> fieldsAndValues){
return Completable.create(emitter -> ref.update(fieldsAndValues)
.addOnSuccessListener(aVoid -> emitter.onComplete())
.addOnCompleteListener(task -> emitter.onComplete())
.addOnFailureListener(emitter::onError));
}
尽管第一个成功执行(调用onSuccess,Firestore中的文档已更新),但包装的代码永远不会从Task到达onSuccess(onSuccess尚未实现,Firestore中的文档从未更新)。但这不会引发任何错误,任务已完成但从未成功。
我不了解这种行为。有什么建议么?我在这里做什么错了?
答案 0 :(得分:2)
您不必一定要对任务使用回调-我认为api在这种意义上是限制性的。
您可以使任务阻塞并在执行后对其进行评估,即
public static Completable updateDocument(DocumentReference ref, Map<String,Object> fieldsAndValues){
return Completable.fromCallable(() -> {
final Task<Void> task = ref.update(fieldsAndValues);
Tasks.await(task);
if(task.isSuccessful()) {
return Completable.complete();
} else {
return Completable.error(new IllegalStateException("Task not successful", task.getException()));
}
});
}
作为您的原始问题,请尝试在闭包中保留对任务的引用,即
public static Completable updateDocument(DocumentReference ref, Map<String, Object> fieldsAndValues) {
return Completable.create(emitter -> {
final Task<Void> task = ref.update(fieldsAndValues)
task.addOnSuccessListener(aVoid -> emitter.onComplete())
.addOnCompleteListener(task -> emitter.onComplete())
.addOnFailureListener(emitter::onError);
});
}
答案 1 :(得分:1)
您subscribe
d是否已完成?
除非您订阅,否则ref.update()
将永远不会执行
updateDocument(...)
.subscribe(...) // <- this
注意:您的第一个代码也有问题。
Completable
天时应执行subscribe
个。但是,在您的第一个代码中,当您调用updateDocument()
时将执行该代码。
例如,如果您执行Completable c = updateDocument(...)
,则不应更新文档,因为您尚未subscribe()
,但是您的第一个代码将更新文档。