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,你可以更灵活地测试响应式流,尤其是当你需要模拟复杂或非标准的数据流时,它们是非常有用的工具。

相关推荐
ye9015 分钟前
银河麒麟V10服务器版 + openGuass + JDK +Tomcat
java·开发语言·tomcat
武昌库里写JAVA17 分钟前
Oracle如何使用序列 Oracle序列使用教程
java·开发语言·spring boot·学习·课程设计
做题不NG1 小时前
大模型应用开发-LangChain4j
java
今天背单词了吗9801 小时前
算法学习笔记:7.Dijkstra 算法——从原理到实战,涵盖 LeetCode 与考研 408 例题
java·开发语言·数据结构·笔记·算法
高兴达2 小时前
RPC--Netty客户端实现
java·spring·rpc
重庆小透明2 小时前
力扣刷题记录【1】146.LRU缓存
java·后端·学习·算法·leetcode·缓存
TTc_3 小时前
@Transactional事务注解的批量回滚机制
java·事务
wei_shuo3 小时前
飞算 JavaAI 开发助手:深度学习驱动下的 Java 全链路智能开发新范式
java·开发语言·飞算javaai
欧阳秦穆4 小时前
apoc-5.24.0-extended.jar 和 apoc-4.4.0.36-all.jar 啥区别
java·jar