动脑学院Rxjava预习资料 RxJava2 响应式编程介绍

RxJava2 响应式编程介绍

  • 响应式编程&RxJava
    • 基本概念
    • 响应式宣言
    • 响应式扩展
    • 响应式流规范
  • RxJava 基础
    • RxJava现状
    • RxJava2 优势
      • 与其它编程模式/库相比
      • RxJava 1 vs RxJava 2
    • RxJava2中的响应式类
      • RxJava2 主要类关系图
      • Flowable & Observable
      • Single & Completable & Maybe
  • RxJava2 的主要操作
    • 创建一个Flowable
    • 转换、过滤与聚合操作
    • 异步与并发(Asynchronized & Concurrency)
      • observeOn & subscribeOn & Scheduler
      • 多线程并发示例
      • 阻塞与非阻塞示例
    • 错误处理 (Error Handling)
    • 冷热数据流
      • ConnectableFlowable & publish & connect
      • replay
      • cache
    • 背压(Backpressure)问题
    • RxJava 测试
  • Reference

响应式编程

基本概念

  • 同步/异步: 关注的是消息通信机制,同步是指 发出一个调用,在没有得到结果之前,该调用就不返回,但是一旦调用返回,就得到返回值了; 而异步是指 调用发出后,调用直接返回,但不会立刻得到调用的结果。而是在调用发出后,被调用者通过状态、通知来通知调用者,或通过回调函数处理这个调用; 异步强调被动通知。

  • 阻塞/非阻塞:关注的是程序在等待调用结果(消息,返回值)时的状态;阻塞调用是指调用结果返回之前,当前线程会被挂起。调用线程只有在得到结果之后才会返回。 非阻塞调用指在不能立刻得到结果之前,该调用不会阻塞当前线程。如java8 stream为阻塞式,Future为非阻塞式的; 非阻塞强调状态主动轮询。

  • 并发(Concurrency)与并行(parallelism): 并发是一个更通用的概念,两个并发线程被调度在一个单核处理上运行,这就是并发性,而非并行. 并行性更可能出现在多核,多CPU或分布式系统上。编写代码时一般不区分两者。

  • 函数式编程Functional Programming):函数式编程将程序描述为表达式和变换,以数学方程的形式建立模型,并尽量避免可变的状态。每个逻辑分类(filter,map,reduce等)都由不同函数所表示,这些实现底层次的变换,而用户定义的高阶函数作为参数来实现真正的业务。

  • 函数响应式编程(Functional Reactive Programming)):响应式编程是建立在观者者模式上的一种编程范式,对异步数据流进行编程,同时该事件流是按时间排序的序列(不同于java8中的stream);虽然响应式编程框架不一定要求是函数式的,但是RxJava等响应式编程框架都是结合了函数式编程的。

  • 其它名词: https://github.com/reactivemanifesto/reactivemanifesto/blob/master/glossary.zh-cn.md

响应式宣言

响应式宣言,Reactive Manifesto: 来自不同领域的组织正在不约而同地发现一些看起来如出一辙的软件构建模式。它们的系统更加稳健,更加有可回复性,更加灵活,并且以更好的定位来满足现代的需求。

响应式宣言针对一个系统而言,并不等同于 响应式编程规范,响应式系统应该满足如下特点:

  • 反应灵敏的[Responsive]:只要有可能,系统就会及时响应。
  • 有回复性的[Resilient]:系统在面临故障时也能保持及时响应。
  • 可伸缩的[Elastic]:系统在变化的工作负载下保持及时响应。
  • 消息驱动的[Message Driven]:响应式系统依赖异步消息传递来建立组件之间的界限,这一界限确保了松耦合,隔离,位置透明性等特性的实现,还提供了以消息的形式把故障委派出去的手段。
动脑学院Rxjava预习资料 RxJava2 响应式编程介绍_第1张图片
QQ截图20180525165803.png

响应式扩展

  • 响应式扩展(Reactive Extensions, 简写为ReactiveX,Rx),最初是LINQ的一个扩展,由微软的架构师Erik Meijer领导的团队开发,在2012年11月开源[1];

  • ReactiveX = Observer Pattern + Iterator Pattern + Functional Programming。Rx 让开发者可以利用可观察序列和LINQ风格查询操作符来编写异步和基于事件的程序;

  • ReactiveX是一个编程模型,目标是提供一致的编程接口,帮助开发者更方便的处理异步数据流,ReactiveX的思想是跨平台的,学习其他语言的基本语法之后,可以做到 “learn once, write everywhere”;

  • Rx近几年越来越流行了,现在已经支持几乎全部的流行编程语言了,Rx的大部分语言库由ReactiveX这个组织负责维护,比较流行的有RxJava/RxJS/Rx.NET,社区网站是 reactivex.io。

  • RxJava 是 ReactiveX 规范的JVM平台实现;

响应式流规范

  • Reactive Streams 规范提供一个非堵塞的异步流处理的背压(backpressure)标准;Reactive Streams的目标是增加抽象层,而不是进行底层的流处理,规范将这些问题留给了库实现来解决。

  • 对于JVM,目前已经有多个库实现该标准,RxJava2, akka-streams(Playing 有使用),Reactor(Spring Reactive有使用) 等;

  • 统一标准的好处就是 各个实现产生的数据可以方便的转换和消费;

示例

    Path filePath = Paths.get("build.gradle");
    // RxJava2 to Reactor
    Flowable flowable = Flowable
            .fromCallable(() -> Files.readAllLines(filePath))
            .flatMap(x -> Flowable.fromIterable(x));
    Flux.from(flowable).count().subscribe(System.out::println);

    // Reactor to RxJava2
    try
    {
        Flux flux = Flux.fromIterable(Files.readAllLines(filePath));
        Flowable.fromPublisher(flux).count()
                .subscribe(System.out::println);
    }
    catch (IOException e)
    {
        e.printStackTrace();
    }
  • Reactive Streams JVM接口由以下四个interface 组成:

    • Publisher : 消息发布者

    • Subscriber : 消息订阅者

    • Subscription : 一个订阅

    • Processor : Publisher + Subscriber 的结合体

  • Reactive Streams 规范主要目标:

    • 通过异步边界(Asynchronous Boundary)来解耦系统组件。 解偶的先决条件,分离事件/数据流的发送方和接收方的资源使用;
    • 为背压( back pressure ) 处理定义一种模型。流处理的理想范式是将数据从发布者推送到订阅者,这样发布者就可以快速发布数据,同时通过压力处理来确保速度更快的发布者不会对速度较慢的订阅者造成过载。背压处理通过使用流控制来确保操作的稳定性并能实现优雅降级,从而提供弹性能力。
  • 该规范将包含在 JDK 9 的java.util.concurrent.Flow 类中,包含四个接口类。

RxJava 基础

RxJava现状

  • RxJava 项目地址 https://github.com/ReactiveX/RxJava
  • RxJava 1.x 先于 Reactive Streams 规范出现,部分接口支持Reactive Streams 规范;
  • RxJava 2.0 于 2016.10.29 正式发布,已经按照Reactive-Streams 规范完全的重写, 基于Java8+;
  • RxJava 2.0已经独立于RxJava 1.x而存在,即 RxJava2(io.reactivex.) 使用与RxJava1(rx.) 不同的包名。
  • RxJava 目前在Android 开发上应用较多;

为什么选择 RxJava2

与其它编程模式/库相比

  • Rx扩展了观察者模式用于支持数据和事件序列,通过一些的操作符号,统一风格,用户无需关注底层的实现:如线程、同步、线程安全、并发数据结构和非阻塞IO;
单个数据 多个数据
同步 T getData() Iterable getData()
异步 Future getData() Observable getData()
  • 对于单条数据可以选择Future 模式,但是多条异步数据组合,Future就相对不方便;

  • 对于同步的多条数据,Observable/Flowable 和 Java8 Stream 都要比Iterable更加方便。

  • 对比 Java 8 Stream,都是属于函数式编程(Monad),Stream主要是对数据集的阻塞处理, 而Rx 是非阻塞的,RxJava 提供更加丰富的操作符。

  • 对比 Java8 Stream, Rx 中的数据流/事件流,扩展到了时间维度,在时间维度上处理发射的数据,所以很多操作符都是与时间有关的。

  • 对比 Reactor-core, 都遵循响应式流规范,Reactor(Flux)和RxJava2(Flowable)可以相互转换, Reactor更多的依赖java8的函数式接口,RxJava2 所有函数式接口都提供异常抛出,在写代码时更加便利。

  • RxJava 函数式风格,简化代码(Rx的操作符通通常可以将复杂的难题简化为很少的几行代码), 异步错误处理,轻松使用并发;

RxJava 1 vs RxJava 2

  • RxJava 2x 不再支持 null 值,如果传入一个null会抛出 NullPointerException
    Observable.just(null);
    Single.just(null);
    Flowable.just(null);
    Maybe.just(null);
    Observable.fromCallable(() -> null)
            .subscribe(System.out::println, Throwable::printStackTrace);
    Observable.just(1).map(v -> null)
            .subscribe(System.out::println, Throwable::printStackTrace);
  • RxJava2 所有的函数接口(Function/Action/Consumer)均设计为可抛出Exception,解决编译异常需要转换问题。

  • RxJava1 中Observable不能很好支持背压,在RxJava2 中将Oberservable实现成不支持背压,而新增Flowable 来支持背压。

  • 详细参考请参考[5].

RxJava2中的响应式类

RxJava2 主要类关系图

如下图所示,为RxJava2中的主要类关系图,可清晰知道各响应式类的联系和区别。后面无特别说明均以Flowable说明。
动脑学院Rxjava预习资料 RxJava2 响应式编程介绍_第2张图片
Publisher-Subscriber-class-relation.png

Flowable & Observable

  • Observable: 不支持背压;

  • Flowable : Observable新的实现,支持背压,同时实现Reactive Streams 的 Publisher 接口。

  • 什么时候用 Observable:

    • 一般处理最大不超过1000条数据,并且几乎不会出现内存溢出;
    • 如果式GUI 鼠标事件,频率不超过1000 Hz,基本上不会背压(可以结合 sampling/debouncing 操作);
    • 如果处理的式同步流而你的Java平台又不支持Java Stream(如果有异常处理,Observable 比Stream也更适合);
  • 什么时候用 Flowable:

    • 处理以某种方式产生超过10K的元素;
    • 文件读取与分析,例如 读取指定行数的请求;
    • 通过JDBC 读取数据库记录, 也是一个阻塞的和基于拉取模式,并且由ResultSet.next() 控制;
    • 网络IO流;
    • 有很多的阻塞和/或 基于拉取的数据源,但是又想得到一个响应式非阻塞接口的。

Single & Completable & Maybe

  • Single: 可以发射一个单独onSuccess 或 onError消息。它现在按照Reactive-Streams规范被重新设计,并遵循协议 onSubscribe (onSuccess | onError)? .SingleObserver改成了如下的接口;
interface SingleObserver {
    void onSubscribe(Disposable d);
    void onSuccess(T value);
    void onError(Throwable error);
}
  • Completable: 可以发送一个单独的成功或异常的信号,按照Reactive-Streams规范被重新设计,并遵循协议onSubscribe (onComplete | onError)?
    Completable.create(new CompletableOnSubscribe()
    {
        @Override
        public void subscribe(CompletableEmitter e) throws Exception
        {
            Path filePath = Paths.get("build.gradle");
            Files.readAllLines(filePath);
            e.onComplete();
        }
    }).subscribe(() -> System.out.println("OK!"),
            Throwable::printStackTrace);
  • Maybe:从概念上来说,它是Single 和 Completable 的结合体。它可以发射0个或1个通知或错误的信号, 遵循协议 onSubscribe (onSuccess | onError | onComplete)?。
Maybe.just(1)
        .map(v -> v + 1)
        .filter(v -> v == 1)
        .defaultIfEmpty(2)
        .test()
        .assertResult(21);
//        java.lang.AssertionError: Values at position 0 differ; Expected: 21 (class: Integer), Actual: 2 (class: Integer) (latch = 0, values = 1, errors = 0, completions = 1)
//
//        at io.reactivex.observers.BaseTestConsumer.fail(BaseTestConsumer.java:133)
//        ....

RxJava2 的主要操作

我们已经知道 RxJava主要特性为为一个扩展的观察者模式、流式操作和异步编程,支持ReactiveX 规范给出的一些操作, 同时RxJava2 符合响应式流规范,接下来以Flowable为例,按照功能分类讲解RxJava2中的重要操作[9];

创建一个Flowable

  • fromArray & fromIterable & just,直接从数组或迭代器中产生;
    List list = Arrays.asList(
            "blue", "red", "green", "yellow", "orange", "cyan", "purple"
    );
    Flowable.fromIterable(list).skip(2).subscribe(System.out::println);
    Flowable.fromArray(list.toArray()).subscribe(System.out::println);
    Flowable.just("blue").subscribe(System.out::println);
  • fromFuture & fromCallable:

    fromFuture, 事件从非主线程中产生; fromCallable, 事件从主线程中产生, 在需要消费时生产;

    ExecutorService executor = Executors.newFixedThreadPool(2);
    System.out.println("MAIN: " + Thread.currentThread().getId());
    Callable callable = () -> {
        System.out.println("callable [" + Thread.currentThread().getId() + "]: ");
        Path filePath = Paths.get("build.gradle");
        return Files.readAllLines(filePath).stream().flatMap(s -> Arrays.stream(s.split
                (""))).count() + "";
    };

    Future future = executor.submit(callable);

    Consumer onNext = v -> System.out
            .println("consumer[" + Thread.currentThread().getId() + "]:" + v);

    Flowable.fromCallable(callable).subscribe(onNext);
    Flowable.fromFuture(future).subscribe(onNext);
    System.out.println("END");
  • fromPublisher ,从标准(Reactive Streams)的发布者中产生;

  • 自定义创建(generate & create)

    下面以斐波那契数列产生为例说明 generate & create的使用, generate为RxJava2新增的创建方式。

    class Fib
    {
        long a;
        long b;

        public Fib(long a, long b)
        {
            this.a = a;
            this.b = b;
        }

        public long fib()
        {
            return a + b;
        }
    }

    //斐波那契数列
    Flowable.create(new FlowableOnSubscribe()
    {
        @Override
        public void subscribe(FlowableEmitter e) throws Exception
        {
            Fib start = new Fib(1L, 1L);

            while (!e.isCancelled()) {
                e.onNext(start);
                start = new Fib(start.b, start.fib());
            }
            e.onComplete();
        }
    }, BackpressureStrategy.BUFFER).map(x -> x.fib()).take(10).subscribe(System.out::println);

    Flowable.generate(() -> new Fib(1L, 1L), (x, y) -> {
        Fib fib = new Fib(x.b, x.fib());
        y.onNext(fib);
        return fib;
    }).ofType(Fib.class).map(x -> x.fib()).take(10).subscribe(System.out::println);
  • amb & concat & merge, 由多个Flowable产生结合;

    • amb: 给定两个或多个Flowable,只发射最先发射数据的Flowable,如下面示例中的f1被发射;

    • concat: 给定多个Flowable, 按照Flowable数组顺序,依次发射数据,不会交错,下面示例中f1,f2中数据依次发射;

    • merge: 给定多个Flowable, 按照Flowable数组中数据发射的顺序组合成新的Flowable,各Flowable数据可能会交错(等价于转换操作中的flatMap);

    • switchOnNext:给定能发射多个Flowable的Flowable,顺序发射各子Flowable,最新发射的子Flowable覆盖当前子Flowable中还未发射的元素(由switchMap实现)。

      动脑学院Rxjava预习资料 RxJava2 响应式编程介绍_第3张图片
      zip

    Flowable f1 = Flowable.intervalRange(1, 10, 1, 1, TimeUnit.SECONDS).map(index -> "f1-" + index);
    Flowable f2 = Flowable.intervalRange(1, 3, 2, 2, TimeUnit.SECONDS).map(index -> "f2-" + index);

    Flowable.ambArray(f1, f2).map(x -> "amb: " + x).subscribe(System.out::println);
    System.out.println("----------concat-----------");
    Flowable.concat(f1, f2).map(x -> "concat: " + x).subscribe(System.out::println);

    System.out.println("----------merge-----------");
    Flowable.merge(f1, f2).map(x -> "merge: " + x).subscribe(System.out::println);

    Flowable[] flowables = new Flowable[]{f1, f2};
    Flowable.switchOnNext(Flowable.intervalRange(0, 2, 0, 3, TimeUnit.SECONDS).map(i -> flowables[i.intValue()]))
            .map(x -> "switchOnNext-" + x).subscribe(System.out::println);
    Flowable.intervalRange(0, 2, 0, 3, TimeUnit.SECONDS).map(i -> flowables[i.intValue()])
            .switchMap((io.reactivex.functions.Function) Functions.identity())
            .map(x -> "switchMap-" + x).subscribe(System.out::println);
  • zip & combineLatest, 多Flowable中元素结合变换

    • zip :每个Flowable中的元素都按顺序结合变换,直到元素最少Flowable的已经发射完毕;

      动脑学院Rxjava预习资料 RxJava2 响应式编程介绍_第4张图片
      zip

    • combineLatest: 每个Flowable中的发射的元素都与其他Flowable最近发射的元素结合变换,知道所有的Flowable的元素发射完毕;

      动脑学院Rxjava预习资料 RxJava2 响应式编程介绍_第5张图片
      zip

转换、过滤与聚合操作

在Java8中Stream也有包含这些功能的操作,由于多了时间这个维度,在 RxJava 中操作相对更加丰富。 这里主要介绍一些重点操作。

  • buffer & groupBy & window

    buffer 和 window 都可以按时间或者元素数量窗口,buffer是直接转换成元素集,window是将元素集转换成另一个Flowable, groupBy,按照key 来分组,需要元素发射完成才能消费,如果只是对数据处理使用Java8 groupBy更方便;

    Flowable f1 = Flowable.intervalRange(1, 10, 1, 1, TimeUnit.SECONDS).delay((t) ->
            Flowable.timer(t % 3 + new Random().nextLong() % 3, TimeUnit.SECONDS))
            .map(index -> index % 3 + "-f1-" + index);
    f1.buffer(5, TimeUnit.SECONDS).map(x -> "buffer-" + x).subscribe(System.out::println);

    f1.window(5, TimeUnit.SECONDS).map(x -> x.toList())
            .subscribe(x -> x.subscribe(System.out::println));

    Disposable b = f1.groupBy((x) -> x.split("-", 2)[0])
            .subscribe(x -> x.toList().subscribe(System.out::println));
    Map> map = f1.toList().blockingGet().stream()
                .collect(Collectors.groupingBy((x) -> x.split
                        ("-", 2)[0]));
    System.out.println(map);

    while (!b.isDisposed()) {
    }
  • debounce & throttleFirst & sample 按照时间区间采集数据

debounce 防抖动,元素发射后在设定的超时时间内没有其它元素发射,则将此元素用于后续处理, 在前端APP应用较多。 如果是空间上的防抖动可以利用distinctUntilChanged操作符。
动脑学院Rxjava预习资料 RxJava2 响应式编程介绍_第6张图片
debounce

throttle 限流操作,对于 throttleFirst是 取发射后元素,经过间隔时间后的第一个元素进行发射。
动脑学院Rxjava预习资料 RxJava2 响应式编程介绍_第7张图片
throttleFirst

sample 数据采样, 对于源数据,发射间隔时间内的最后出现的元素。
动脑学院Rxjava预习资料 RxJava2 响应式编程介绍_第8张图片
sample
  • take & skip & first & emlmentAt,精确获取数据(集)

    take, 类似java8 limit 操作,但是这里支持更多的操作(take/takeLast/takeUntil/takeWhen),同时支持在时间区间上获取数据集; skip, 类似java8 skip 操作,但是这里的可以扩展到时间区间上 first/firstElement/last/lastElement, 由 Flowable -> Single/Maybe.

        Flowable f1 = Flowable
                .fromArray("blue", "red", "green", "yellow11", "orange", "cyan", "purple"
                );

        f1.elementAt(4, "hello").subscribe(System.out::println);
        //out: orange
        f1.takeUntil(x -> x.length() > 5).map(x -> "takeUntil-" + x).toList()
                .subscribe(System.out::println);
        //out: [takeUntil-blue, takeUntil-red, takeUntil-green, takeUntil-yellow11]
        f1.takeWhile(x -> x.length() <= 5).map(x -> "takeWhile-" + x).toList()
                .subscribe(System.out::println);
        //out: [takeWhile-blue, takeWhile-red, takeWhile-green]

        f1.skipWhile(x -> x.length() <= 5).map(x -> "skipWhile-" + x).toList()
                .subscribe(System.out::println);
        //[skipWhile-yellow11, skipWhile-orange, skipWhile-cyan, skipWhile-purple]

        Disposable d = f1.delay(v -> Flowable.timer(v.length(), TimeUnit.SECONDS))
                .skipUntil(Flowable.timer(5, TimeUnit.SECONDS)).map(x -> "skipUntil-" + x)
                .subscribe(System.out::println);
//        skipUntil-green
//        skipUntil-orange
//        skipUntil-purple
//        skipUntil-yellow11
        while (!d.isDisposed()) {
        }

异步与并发(Asynchronized & Concurrency)

RxJava 通过一些操作统一了 同步和异步,阻塞与非阻塞,并行与并发编程。

observeOn & subscribeOn & Scheduler

  • subscribeOn 和 observeOn 都是用来切换线程用的,都需要参数 Scheduler.

  • Scheduler ,调度器, 是RxJava 对线程控制器 的 一个抽象,RxJava 已经内置了几个 Scheduler ,它们已经适合大多数的使用场景:

    • trampoline, 直接在当前线程运行(继续上一个操作中,最后处理完成的数据源所处线程,并不一定是主线程),相当于不指定线程;

    • computation, 这个 Scheduler 使用的固定的线程池(FixedSchedulerPool),大小为 CPU 核数, 适用于CPU 密集型计算。

    • io,I/O 操作(读写文件、读写数据库、网络信息交互等)所使用的 Scheduler。行为模式和 newThread() 差不多,区别在于 io() 的内部实现是是用一个无数量上限的线程池,可以重用空闲的线程,因此多数情况下 io() 比 newThread() 更有效率;

    • newThread, 总是启用新线程,并在新线程中执行操作;

    • single, 使用定长为1 的线程池(newScheduledThreadPool(1)),重复利用这个线程;

    • Schedulers.from, 将java.util.concurrent.Executor 转换成一个调度器实例。

      java.util.function.Consumer pc = x -> System.out
             .println("Thread[" + Thread.currentThread().getName() + " ," + Thread
                     .currentThread().getId() + "] :" + x);
      Executor executor = Executors.newFixedThreadPool(2);
      Schedulers.from(executor).scheduleDirect(() -> pc.accept("executor one"));
      Schedulers.from(executor).scheduleDirect(() -> pc.accept("executor two"));
      Schedulers.trampoline().scheduleDirect(() -> pc.accept("trampoline"), 1, TimeUnit.SECONDS);
      Schedulers.single().scheduleDirect(() -> pc.accept("single one DONE"));
      Schedulers.single().scheduleDirect(() -> pc.accept("single two DONE"));
      Schedulers.computation()
             .scheduleDirect(() -> pc.accept("computation one DONE"), 1, TimeUnit.SECONDS);
      Schedulers.computation()
             .scheduleDirect(() -> pc.accept("computation two DONE"), 1, TimeUnit.SECONDS);
      Schedulers.io().scheduleDirect(() -> pc.accept("io one DONE"));
      Schedulers.io().scheduleDirect(() -> pc.accept("io two DONE"), 1, TimeUnit.SECONDS);
      Schedulers.io().scheduleDirect(() -> pc.accept("io tree DONE"), 1, TimeUnit.SECONDS);
      Schedulers.newThread().scheduleDirect(() -> pc.accept("newThread tree DONE"));
       
       
      
    • subscribeOn 将Flowable 的数据发射 切换到 Scheduler 所定义的线程, 只有第一个 subscribeOn 操作有效 ;

    • observeOn 指定 observeOn 后续操作所在线程,可以联合多个 observeOn 将切换多次 线程 ;

      示例 Schedulers.newThread() 定义的线程发送数据;

      Schedulers.computation() 定义的线程 执行doOnNext;

      Schedulers.single() 执行 subscribe

        Consumer threadConsumer = x -> System.out
                .println("Thread[" + Thread.currentThread().getName() + " ," + Thread
                        .currentThread().getId() + "] :" + x);
      
        Flowable f1 = Flowable.create((FlowableEmitter e) -> {
            Path dir = Paths.get("/home/clouder/berk/workspaces/cattle").toRealPath();
            try (DirectoryStream dirStream = Files.newDirectoryStream(dir)) {
                Iterator iter = dirStream.iterator();
                while (iter.hasNext() && !e.isCancelled()) {
                    e.onNext(iter.next());
                }
                e.onComplete();
            }
        }, BackpressureStrategy.BUFFER);
        f1.subscribeOn(Schedulers.newThread()).subscribeOn(Schedulers.io())
                .observeOn(Schedulers.computation()).take(5).doOnNext(consumer).observeOn(Schedulers
                .single()).subscribe(consumer);
       
      
      

      多线程并发示例

      上小节给出示例 发射元素都会经过同样的线程切换,元素间不会产生并行执行的效果。 如果需要达到 类似 Java8 parallel 执行效果。可以采用FlatMap 变换 自定义并发操作,在返回的Flowable进行线程操作,如下示例所示:

      • f1 中元素会在Schedulers.newThread()中发射;

      • 读取文本内容的操作(Files::readAllLines, Collection::size) 会在 Schedulers.io() 所指定的线程池执行;

      • sorted 操作会在 Schedulers.computation() 所指定的线程池中执行;

      • subscribe() 同样会在 Schedulers.computation() 所指定的线程池中执行;

          f1.filter(Files::isRegularFile).doOnNext(consumer).subscribeOn(Schedulers.newThread())
                  .flatMap(y -> Flowable.just(y).subscribeOn(Schedulers.io())
                          .map(Files::readAllLines)).map(Collection::size)
                  .observeOn(Schedulers.computation()).doOnNext(consumer)
                  .sorted(Comparator.naturalOrder())
                  .observeOn(Schedulers.trampoline()).subscribe(consumer);
      

      阻塞与非阻塞示例

      • 从阻塞到非阻塞 我们可以通过 subscribeOn() 来达到;

         //f1 为 主线程发射数据的Flowable
         //会阻塞主线程知道消费完成
         f1.subscribe(System.out::println);
        
         // d会理解返回.
         Disposable d = f1.subscribeOn(Schedulers.newThread())
                 .subscribe(System.out::println);
         while (!d.isDisposed()) {
         }
        
        • 从非阻塞到阻塞,可以通过blocking* 相关操作来实现
          Flowable f2 = f1.subscribeOn(Schedulers.newThread());
          //f2 为非阻塞flowable
          // 可以通过 blockingSubscribe 变为在主线程上消费
          f2.blockingSubscribe(System.out::println);
          // 也可以通过下面操作返回结果。
          List list = f2.toList().blockingGet();
          Iterable iterator = f2.blockingIterable();
      

      错误处理 (error handling)

      • RxJava

      • 无需显示的catch 编译异常,RxJava2 已经支持所有函数接口抛出Exception;

      • 更方便在异步多线程环境下进行错误处理;

        如下示例 会打印第一个异常;

          Flowable f1 = Flowable.interval(500, TimeUnit.MILLISECONDS).map(index -> {
              throw new IOException(index + "");
          }).map(index -> {
              throw new IllegalArgumentException(index + "");
          });
          Disposable d = f1.subscribe(System.out::println, Throwable::printStackTrace);
          while (!d.isDisposed()) {
          }
      
      • 异常可以被转换,源数据发射终止
        • 当出现异常时,可以通过 onErrorReturn* 转换成一个正常值返回;
        • 当出现异常时,通过 onErrorResumeNext 自定义一个Publisher返回,意味着可以转换一个异常类型;
          Flowable f1 = Flowable.interval(500, TimeUnit.MILLISECONDS).map(index -> {
              throw new IOException(index + "");
          }).map(index -> {
              throw new IllegalArgumentException(index + "");
          });
          f1.onErrorReturnItem(-1L).take(5)
                  .subscribe(System.out::println, Throwable::printStackTrace);
          // 打印 -1 
          Disposable d = f1.onErrorResumeNext(e -> {
              if (e instanceof IOException) {
                  return Flowable.error(new UncheckedIOException((IOException) e));
              }
              return Flowable.error(e);
          }).subscribe(System.out::println, Throwable::printStackTrace);
          // 打印 UncheckedIOException 异常
          while (!d.isDisposed()) {
          }
      
      • Flowable,map抛出异常,但数据继续发射

        暂没有找到直接方法可以达到,但可以采取如下两种方法达到

          Function exceptionMap = x -> {
              if (new Random().nextInt(5) > 2) {
                  throw new IOException(x + "");
              }
              return x;
          };
      
          // 使用flatMap + onErrorReturnItem
          Flowable f1 = Flowable.interval(500, TimeUnit.MILLISECONDS);
          f1.flatMap(index -> Flowable.just(index).map(exceptionMap).onErrorReturnItem(-1L))
          .take(5).subscribe(System.out::println);
      
         //直接封装lift 操作
         public class ErrorResumeOperator implements FlowableOperator
          {
              private final Function function;
              private final D defaultValue;
      
              public ErrorResumeOperator(Function function, D defaultValue)
              {
                  this.function = function;
                  this.defaultValue = defaultValue;
              }
      
              @Override
              public Subscriber apply(Subscriber observer) throws Exception
              {
                  Subscriber subscriber = new Subscriber()
                  {
                      @Override
                      public void onSubscribe(Subscription s)
                      {
                          observer.onSubscribe(s);
                      }
      
                      @Override
                      public void onNext(U onNext)
                      {
                          try {
                              observer.onNext(function.apply(onNext));
                          }
                          catch (Exception e) {
                              observer.onNext(defaultValue);
                          }
                      }
      
                      @Override
                      public void onError(Throwable t)
                      {
                          observer.onError(t);
                      }
      
                      @Override
                      public void onComplete()
                      {
                          observer.onComplete();
                      }
                  };
                  return subscriber;
              }
          }
          Disposable d = f1.lift(new ErrorResumeOperator<>(exceptionMap, -1L)).take(5)
                  .subscribe(System.out::println);
      
          while (!d.isDisposed()) {
          }
      
      • 出错重试(retry)

        RxJava 提供了retry以及相关的多个操作,提供出错后重新发射数据功能;

      
          Function exceptionMap = x -> {
              if (new Random().nextInt(5) > 3) {
                  throw new IOException(x + "");
              }
              if (new Random().nextInt(6) < 1) {
                  throw new SQLException(x + "");
              }
              return x;
          };
          Flowable f1 = Flowable.interval(500, TimeUnit.MILLISECONDS);
      // 仅为 IOException 异常时最多重试3次,其它异常立即打印异常
          Disposable d = f1.map(exceptionMap).retry(3, e -> e instanceof IOException)
                  .subscribe(System.out::println, Throwable::printStackTrace);
          while (!d.isDisposed()) {
          }
      

      冷热数据流

      ConnectableFlowable & publish & connect

      • ConnectableFlowable 可连接的Flowable, 不管是否消费,只有调用了connect, 数据就一直在发射,不受消费影响 ('冷' 的Flowable 变成'热'的)
      • publish 将 普通 Flowable,变成 ConnectableFlowable ;
      动脑学院Rxjava预习资料 RxJava2 响应式编程介绍_第9张图片
      publishConnect

      ;

          ConnectableFlowable f1 = Flowable.generate(() -> new BufferedReader(new InputStreamReader(System.in))
                  , (reader, e) -> {
                      while (true) {
                          String line = reader.readLine();
                          if (line == null || line.equalsIgnoreCase("exit")) {
                              break;
                          }
                          e.onNext(line);
                      }
                      e.onComplete();
                  }).ofType(String.class).subscribeOn(Schedulers.io()).publish();
      
          TimeUnit.SECONDS.sleep(5);
          f1.connect(System.out::println);
          TimeUnit.SECONDS.sleep(5);
          f1.observeOn(Schedulers.newThread()).map(x -> "s0- " + x).subscribe(System.out::println);
          TimeUnit.SECONDS.sleep(5);
          f1.map(x -> "s1- " + x).subscribe(System.out::println);
          TimeUnit.SECONDS.sleep(50);
      

      replay

      replay 将Flowable变成 ConnectableFlowable, 在connect之后,确保每次消费都使用相同数据。

          java.util.function.Function> m = s -> v -> System.out
                  .println("[" + System.currentTimeMillis() / 100 + "] " + s + "-" + v);
          ConnectableFlowable f1 = Flowable.intervalRange(1, 100, 0, 1, TimeUnit.SECONDS)
                  .onBackpressureBuffer().replay();
          m.apply("").accept("start");
          TimeUnit.SECONDS.sleep(5);
          f1.connect();
          TimeUnit.SECONDS.sleep(5);
          f1.subscribe(m.apply("o1"));
      
          TimeUnit.SECONDS.sleep(5);
          f1.subscribe(m.apply("o2"));
          TimeUnit.SECONDS.sleep(20);
      

      cache

      缓存功能,将Flowable进行缓存

          java.util.function.Function> m = s -> v -> System.out
                  .println("[" + System.currentTimeMillis() / 100 + "] " + s + "-" + v);
      
          Flowable f1 = Flowable.create((FlowableEmitter e) -> {
              Path dir = Paths.get("/home/clouder/berk/workspaces/cattle").toRealPath();
              try (DirectoryStream dirStream = Files.newDirectoryStream(dir)) {
                  Iterator iter = dirStream.iterator();
                  while (iter.hasNext() && !e.isCancelled()) {
                      Path path = iter.next();
                      m.apply("-----create").accept(path);
                      e.onNext(path);
                  }
                  e.onComplete();
              }
          }, BackpressureStrategy.BUFFER).cache();
      
          f1.count().subscribe(m.apply("count"));
          f1.filter(Files::isDirectory).subscribe(m.apply("filter"));
      

      背压(Backpressure)问题

      问题描述: 在rxjava中会经常遇到一种情况就是被观察者发送消息太快以至于它的操作符或者订阅者不能及时处理相关的消息。那么随之而来的就是如何处理这些未处理的消息。

      如下示例: f1 比 f2 元素发射速度快一倍。而zip是按照发射顺序结合,所以出现f1的产生速度快于其消费速度,因此会有背压问题产生(当发射到一定数量时会有异常抛出)。

          Consumer consumer = v -> System.out.println("[" + System.currentTimeMillis() / 100 + "] " + v);
          Flowable f1 = Flowable.interval(100, TimeUnit.MILLISECONDS);
          Flowable f2 = Flowable.interval(200, TimeUnit.MILLISECONDS);
      
          Flowable f3 = Flowable.zip(f1, f2, (x, y) -> x * 10000 + y);
          f3.subscribe(consumer);
      
      

      对于出现的背压问题: - Flowable默认队列大小为128,并且规范要求,所有的操作符强制支持背压。 - 通过操作节流(Throttling)相关操作(sample 、throttleLast、throttleFirst、throttleWithTimeout、debounce等)来改变Flowable的发射数率;

      • 通过设置缓冲区和窗口(buffer,window)操作,来缓存过剩的数据,然后发送特定数据。

      • 设置背压策略(onBackpressurebuffer & onBackpressureDrop & onBackpressureLatest)

      RxJava 测试

      RxJava2 支持test() 操作符,将Flowable转变为 TestSubscriber,从而支持多种断言操作。

          List list = Arrays.asList(
                  "orange", "blue", "red", "green", "yellow", "cyan", "purple");
      
          Flowable.fromIterable(list).subscribeOn(Schedulers.newThread()).sorted().test().assertValues(list.stream().sorted().toArray(String[]::new));
          Flowable.fromIterable(list).count().test().assertValue(Integer.valueOf(list.size()).longValue());
          List out1 = Flowable.fromIterable(list).sorted().test().values();
      

      Reference

      1. 响应式宣言.https://github.com/reactivemanifesto/reactivemanifesto/blob/master/README.zh-cn.md

      2. RxJava 2.0 Released with Support for Reactive Streams Specification. https://www.infoq.com/news/2016/11/rxjava-2-with-reactive-streams

      3. https://www.lightbend.com/blog/7-ways-washing-dishes-and-message-driven-reactive-systems

      4. Use reactive streams API to combine akka-streams with rxJava. http://www.smartjava.org/content/use-reactive-streams-api-combine-akka-streams-rxjava

      5. What's different in 2.0. https://github.com/ReactiveX/RxJava/wiki/What's-different-in-2.0

      6. Learning Reactive Programming with Java8. https://github.com/zouzhberk/rxjava-study/raw/master/docs/LearningReactiveProgramming.pdf

      示例代码位于: https://github.com/zouzhberk/rxjava-study

      你可能感兴趣的:(动脑学院Rxjava预习资料 RxJava2 响应式编程介绍)