Как обрабатывать исключения в map() в Observable в RxJava
Я хочу сделать это:
Observable.just(bitmap)
.map(new Func1<Bitmap, File>() {
@Override
public File call(Bitmap photoBitmap) {
//File creation throws IOException,
//I just want it to hit the onError() inside subscribe()
File photoFile = new File(App.getAppContext().getCacheDir(), "userprofilepic_temp.jpg");
if(photoFile.isFile()) {//delete the file first if it exists otherwise the new file won't be created
photoFile.delete();
}
photoFile.createNewFile(); //saves the file in the cache dir
FileOutputStream fos = new FileOutputStream(photoFile);
photoBitmap.compress(Bitmap.CompressFormat.JPEG, 90, fos);//jpeg format
fos.close();
return photoFile;
}
})
.subscribe(//continue implementation...);
В основном в методе call()
он может генерировать исключения. Как заставить Observer обрабатывать его в onError()
. Или это не правильный способ подумать об этом?
Ответы
Ответ 1
rx всегда будет ловить ошибку, даже если это RuntimeException.
Таким образом, вы можете бросить какое-то исключение Runtime в catch catch. Вот как это работает.
Observable.just(bitmap)
.map(b -> {
try {
// do some work which throws IOException
throw new IOException("something went wrong");
} catch (IOException e) {
throw new RXIOException(e);
// Or you can use
throw Exceptions.propagate(e);
// This helper method will wrap your exception with runtime one
}
}).subscribe(o -> {
// do something here
}, exception -> exception.printStackTrace());
public static class RXIOException extends RuntimeException {
public RXIOException(IOException throwable) {
super(throwable);
}
}
Ответ 2
С 1.0.15 существует метод fromCallable
factory, который позволяет запускать экземпляр Callable
для каждого абонента, в котором вы также можете отмечать проверенные исключения:
Observable.fromCallable(() -> {
File photoFile = new File(App.getAppContext().getCacheDir(),
"userprofilepic_temp.jpg");
if (photoFile.isFile()) {
//delete the file if it exists otherwise the new file won't be created
photoFile.delete();
}
photoFile.createNewFile(); //saves the file in the cache dir
FileOutputStream fos = new FileOutputStream(photoFile);
photoBitmap.compress(Bitmap.CompressFormat.JPEG, 90, fos);//jpeg format
fos.close();
return photoFile;
})
.subscribe(...)
Edit:
source.flatMap(v -> {
try {
//...
return Observable.just(result);
} catch (Exception e) {
return Observable.error(e);
}
})
.subscribe(...);
Ответ 3
Только что созданный вспомогательный класс для извлечения этого шаблона в другое место:
public class RxRethrow {
public static <T, R> Func1<T, R> rethrow(Func1R<T, R> catchedFunc) {
return t -> {
try {
return catchedFunc.call(t);
} catch (Exception e) {
throw Exceptions.propagate(e);
}
};
}
public interface Func1R<T, R> extends Function {
R call(T t) throws Exception;
}
}
Вы можете называть это следующим образом:
.map(RxRethrow.rethrow(products -> mapper.writer(schema).writeValueAsString(products)))
Ответ 4
Я не знаю, как сложилась ситуация, когда этот вопрос был впервые задан и ответил, но RxJava в настоящее время содержит вспомогательный метод для этой конкретной цели:
Exceptions.propagate(Throwable t)
RxJava Javadoc
Удобный метод для автоматического исключения RuntimeException и Error или переноса любого другого типа исключения в исключение RuntimeException.