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

相关推荐
BD_Marathon几秒前
【IDEA】Debug(调试)
java·ide·intellij-idea
嘟嘟w13 分钟前
JVM性能调优
java
Godson_beginner15 分钟前
Sa-Token (java权限认证框架)
java·开发语言
头发那是一根不剩了16 分钟前
Spring Boot「多数据源并存」的设计思路,它与动态数据源又有什么区别?
java·spring boot·后端
o***592720 分钟前
spring注入static属性
java·后端·spring
风象南24 分钟前
Spring Boot实现HTTPS双向认证
java·spring boot·后端
青春不流名32 分钟前
Java List初始化的例子
java·windows·list
4***172732 分钟前
【MySQL篇】使用Java操作MySQL实现数据交互
java·mysql·交互
sheji341634 分钟前
【开题答辩全过程】以 基于Spring Boot的流浪动物救助系统设计为例,包含答辩的问题和答案
java·spring boot·后端
W***r2641 分钟前
VScode 开发 Springboot 程序
java·spring boot·后端