[英]RxJava: how to process Flowable interactively from console
我创建了一个解析文件的 Flowable (RxJava v3)。 我希望它支持背压。 这很重要,因为文件可能非常大,我不希望它们一次加载到 memory 中。 这是我的第一次尝试:
public Flowable<Byte> asFlowable(InputStream is) {
return Flowable.create(new FlowableOnSubscribe<Byte>() {
@Override
public void subscribe(FlowableEmitter<Byte> emitter) throws Exception {
try (DataInputStream inputStream = new DataInputStream(is)){
if (inputStream.readInt() != SOME_CONSTANT) {
throw new IllegalArgumentException("illegal file format");
}
if (inputStream.readInt() != SOME_OTHER_CONSTANT) {
throw new IllegalArgumentException("illegal file format");
}
final int numItems = inputStream.readInt();
for(int i = 0; i < numItems; i++) {
if(emitter.isCancelled()) {
return;
}
emitter.onNext(inputStream.readByte());
}
emitter.onComplete();
} catch (Exception e) {
emitter.onError(e);
}
}
}, BackpressureStrategy.BUFFER);
}
我使用Flowable.create
而不是Flowable.generate
的原因是因为我需要验证文件,如果文件开头的某些幻数错误或找不到,则抛出错误。 这不适合Flowable.generate
lambdas(但如果您知道更好的方法,请发布)。
好的,让我们假设冷的 Flowable 支持背压。 现在我想在类似控制台的应用程序中处理它。
问题:我想从 Flowable 请求一个新的字节,并在用户每次按下空格时将其打印到控制台(类似于 Linux 中的more
less
操作)。 最好的方法是什么? 我打算直接在public static void main
方法中观察 flowable,因为我需要使用控制台进行读写。
我一直在阅读RxJAva 的 Wiki 中的 Backpressure 部分并找到了这个片段:
someObservable.subscribe(new Subscriber<t>() {
@Override
public void onStart() {
request(1);
}
@Override
public void onCompleted() {
// gracefully handle sequence-complete
}
@Override
public void onError(Throwable e) {
// gracefully handle error
}
@Override
public void onNext(t n) {
// do something with the emitted item "n"
// request another item:
request(1);
}
});
但这让我更加困惑,因为request
方法在 RxJava 3 中似乎不存在。
使用generate
, blockingSubscribe
并从控制台读取一行:
class State {
DataInputStream inputStream;
int count;
int i;
}
BufferedReader bin = new BufferedReader(new InputStreamReader(System.in));
Flowable.generate(() -> {
State s = new State();
s.inputStream = new DataInputStream(is);
try {
if (s.inputStream.readInt() != SOME_CONSTANT) {
throw new IllegalArgumentException("illegal file format");
}
if (s.inputStream.readInt() != SOME_OTHER_CONSTANT) {
throw new IllegalArgumentException("illegal file format");
}
s.count = s.inputStream.readInt();
} catch (IOException ex) {
s.inputStream.close();
throw ex;
}
return s;
}, (state, emitter) -> {
if (state.i < s.count) {
emitter.onNext(state.inputStream.readByte());
s.i++;
}
if (state.i >= s.count) {
emitter.onComplete();
}
}, state -> {
state.inputStream.close();
})
.subscribeOn(Schedulers.io())
.blockingSubscribe(b -> {
System.out.println(b);
bin.readLine();
}, Flowable.bufferSize());
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.