RxJava。将文件读取为可观察的

时间:2017-04-16 21:45:25

标签: java java-8 rx-java reactive-programming rx-java2

我对RxJava和反应式编程完全陌生。 我有一个作业,我必须读取文件并将其存储到Observable。我试图在内部使用BufferedReader创建一个Callable,并使用Observable.fromCallable(),但它没有多大用处。

你能告诉我我该怎么办?

我正在使用RxJava 2.0。

3 个答案:

答案 0 :(得分:3)

一个基本的解决方案,我使用嵌套类FileObservableSource来生成数据,然后推迟创建Observable直到Observer订阅:

import io.reactivex.Observable;
import io.reactivex.ObservableSource;
import io.reactivex.Observer;

import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;

public class StackOverflow {

    public static void main(String[] args) {
        final Observable<String> observable = Observable.defer(() -> new FileObservableSource("pom.xml"));
        observable.subscribe(
                line -> System.out.println("next line: " + line),
                        Throwable::printStackTrace,
                        () -> System.out.println("finished")
                );
    }

    static class FileObservableSource implements ObservableSource<String> {

        private final String filename;

        FileObservableSource(String filename) {
            this.filename = filename;
        }

        @Override
        public void subscribe(Observer<? super String> observer) {
            try {
                Files.lines(Paths.get(filename)).forEach(observer::onNext);
                observer.onComplete();
            } catch (IOException e) {
                observer.onError(e);
            }
        }
    }
}

答案 1 :(得分:2)

对于Java 8,BufferedReader具有文件流和迭代器,可与RxJava一起使用。 https://dzone.com/articles/java-8-stream-rx-java

您可以使用以下代码获得Flowable

 Flowable.fromIterable(bufferedReader.lines()::iterator)

我使用Flowable和Single更加简洁,它仍然可以与Observable一起使用。 该代码段是如何从缓冲读取器读取每一行的一个示例。

try (
       FileReader fr = new FileReader("./folder1/source.txt");
       BufferedReader br = new BufferedReader(fr); 

       //e.g. i write the output into this file
       FileWriter fw = new FileWriter("./folder1/destination.txt");
       BufferedWriter bw = new BufferedWriter(fw)
        ) { 

      //=========================================
      //calling br.lines() gives you the stream. 
      //br.lines()::iterator returns you the iterable
      //=========================================
      Flowable.fromIterable(br.lines()::iterator)

         //at this point you can do what you want for each line
                 //e.g. I split long strings into smaller Flowable<String> chunks.
                .flatMap(i -> splitLine(i))
                 //e.g. I then use the output to query an external server with retrofit.
                 //     which returns me a Single<String> result
                .flatMapSingle(i -> handlePinyin(pr, i))  

                .subscribe(
                        //I save the results from server to destination.txt
                        py -> appendToFile(bw, py),
                        //print and quit if error
                        err -> print("ERROR " + err), 
                        () -> print("Completed!"));


}catch (IOException e) {
        print("Error " + e);
}

答案 2 :(得分:0)

您可以执行与this implementation类似的操作。

然后以这种方式使用here该类:

public static Observable<byte[]> from(InputStream is, int size) {
     return Observable.create(new OnSubscribeInputStream(is, size));
}

最终你可以使用它:

Observable<byte[]> chunks = Bytes.from(file, chunkSize);

更多详情here