Reactor操作符的共享与复用

在 Reactor 中,transformtransformDeferred 是两个用于代码复用和操作符链封装的高级操作符。它们允许你将一组操作符封装成一个函数,并在适当的时候应用到响应式流中。以下是它们的详细总结:


1. transform 操作符

  • 作用transform 操作符允许你将操作符链的一部分封装成一个函数,并在实例化时(assembly time)应用到原始操作符链上。这意味着所有订阅者都会应用相同的操作符链。

  • 适用场景:适用于所有订阅者共享相同行为的场景,例如统一的日志记录、转换等。

  • 示例

    java 复制代码
      // 定义一个函数,用于过滤掉"orange"颜色并将其余颜色转换为大写
      Function<Flux<String>, Flux<String>> filterAndMap =
              f -> f.filter(color -> !color.equals("orange"))
                      .map(String::toUpperCase);
      
      // 从列表创建Flux,并在过滤和映射前打印每个元素
      Flux.fromIterable(Arrays.asList("blue", "green", "orange", "purple"))
              .doOnNext(System.out::println)
              // 使用之前定义的函数对Flux进行转换
              .transform(filterAndMap)
              // 订阅转换后的Flux,并打印每个元素
              .subscribe(d -> System.out.println("Subscriber to Transformed MapAndFilter: " + d));

    执行结果如下

    txt 复制代码
    blue
    Subscriber to Transformed MapAndFilter: BLUE
    green
    Subscriber to Transformed MapAndFilter: GREEN
    orange
    purple
    Subscriber to Transformed MapAndFilter: PURPLE

2. transformDeferred 操作符

  • 作用transformDeferred 操作符与 transform 类似,也允许你将操作符链封装成一个函数,但主要的区别在于函数的执行时机transformDeferred 是在订阅发生时(subscription time)才应用函数的。这意味着它可以为每个订阅者生成不同的操作链,甚至可以维护一些状态(例如计数器、随机数等)。

  • 适用场景:适用于每个订阅者需要不同操作链的场景,例如根据订阅者的身份、时间、状态等动态生成不同的操作链。

  • 示例

    java 复制代码
      // 使用AtomicInteger来实现线程安全的计数器
      AtomicInteger ai = new AtomicInteger();
      
      // 定义一个函数,根据计数器的值来选择不同的处理逻辑
      Function<Flux<String>, Flux<String>> filterAndMap = f -> {
          // 如果计数器的值为1,则过滤掉"orange"颜色,否则过滤掉"purple"颜色
          if (ai.incrementAndGet() == 1) {
              return f.filter(color -> !color.equals("orange"))
                      .map(String::toUpperCase);
          }
          return f.filter(color -> !color.equals("purple"))
                  .map(String::toUpperCase);
      };
      
      // 创建一个Flux,从颜色列表开始,通过.doOnNext()打印每个元素,然后通过.transformDeferred()应用filterAndMap函数
      Flux<String> composedFlux =
              Flux.fromIterable(Arrays.asList("blue", "green", "orange", "purple"))
                      .doOnNext(System.out::println)
                      .transformDeferred(filterAndMap);
      
      // 订阅第一个处理流程的结果
      composedFlux.subscribe(d -> System.out.println("Subscriber 1 to Composed MapAndFilter :" + d));
      // 订阅第二个处理流程的结果
      composedFlux.subscribe(d -> System.out.println("Subscriber 2 to Composed MapAndFilter: " + d));

    执行结果如下

    复制代码
    blue
    Subscriber 1 to Composed MapAndFilter :BLUE
    green
    Subscriber 1 to Composed MapAndFilter :GREEN
    orange
    purple
    Subscriber 1 to Composed MapAndFilter :PURPLE
    blue
    Subscriber 2 to Composed MapAndFilter: BLUE
    green
    Subscriber 2 to Composed MapAndFilter: GREEN
    orange
    Subscriber 2 to Composed MapAndFilter: ORANGE
    purple

3. transformtransformDeferred 的区别

操作符 执行时机 是否为每个订阅者生成不同操作链 是否可以维护状态
transform 实例化时
transformDeferred 订阅时

4. 总结

  • transform:适用于所有订阅者共享相同操作链的场景,例如统一的日志记录、转换等。
  • transformDeferred:适用于每个订阅者需要不同操作链的场景,例如根据订阅者的身份、时间、状态等动态生成不同的操作链。

通过使用 transformtransformDeferred,你可以更灵活地测试响应式流,尤其是当你需要模拟复杂或非标准的数据流时,它们是非常有用的工具。

相关推荐
yaoxin5211234 小时前
434. Java 日期时间 API - Period 基于日期的时间段
java·开发语言·python
何极光5 小时前
IDEA集成Maven
java·maven·intellij-idea
程序员二叉5 小时前
【JUC】ThreadLocal底层原理|内存泄漏|弱引用|跨线程传递方案
java·开发语言·面试·职场和发展·juc
程序员二叉5 小时前
【JUC】线程池全套深度详解|参数|流程|拒绝策略|调优|异常处理
java·开发语言·jvm·算法·面试·juc
老马识途2.06 小时前
在AI的帮助下理解spring的启动过程
java·前端·spring
青山木6 小时前
Hot 100 --- 轮转数组
java·数据结构·算法
Qt程序员6 小时前
掌握 Linux 内核调度:从原理到实现(进程篇)
java·开发语言
code bean6 小时前
【LangChain】检索器完全指南:从向量检索到生产级 RAG 架构
java·开发语言·微服务
大白菜和MySQL6 小时前
java应用排查高线程
java·python
KobeSacre7 小时前
ReentrantLock源码
java