java - 如何使用 Spring Integration Aggregator 进行 'look-ahead' 发布?

标签 java spring spring-integration

问题

我正在使用 Spring Integration(版本 5.0.0.M2,因为我需要动态流注册功能)开发一个应用程序,并且在某些时候我必须根据消息的 correlationId 聚合消息 header 。 header 标记了消息中的一些连续的子序列,即,它在新的子序列开始时发生变化。聚合器的目的是将子序列转换为单独的“分组”消息。
消息来源是外部的且不可预测,因此它无法为发出的消息提供 sequenceSize header 。因此,只有在下一条消息带有另一个 correlationId header (或者在 timeout 毫秒内没有下一条消息)时,才能决定是否释放当前的累积组。表示输入结束)。部分释放是 Not Acceptable 。

问题是 Spring Integration ReleaseStrategy 被设计为仅释放当前累积组(包括当前消息),而我的任务需要““向前看”下一条消息,以确定当前组是否完整。从下一条消息的角度来看,同样的问题可能会被认为是“回望”发布。

所以问题是:有没有办法根据下一条消息的 header (不包括后者)从聚合器中释放消息组?

尝试解决方案

我研究了org.springframework.integration.aggregator.AbstractCorrelatedMessageHandler的源代码,目前我发现的唯一解决方法是一个相当肮脏的技巧 - 我继承自AggregatingMessageHandler 并重写 handleMessageInternal 方法。在该方法中,我将当前消息的correlationId 与上一次调用时保存的correlationId 进行比较。如果不同,我从存储中提取上一个组并用它调用 forceComplete 。然后(以任何方式)我将当前消息处理委托(delegate)给父级的 handleMessageInternal 方法。这是代码片段:

  @Override
  protected void handleMessageInternal(Message<?> message) throws Exception {
    Long currentGroupId = message.getHeaders().get(CORRELATION_ID, Long.class);

    boolean needToReleasePreviousGroup = ((previousGroupId != null) && !previousGroupId.equals(currentGroupId));
    if (needToReleasePreviousGroup) {
      MessageGroup previousGroup = getMessageStore().getMessageGroup(previousGroupId);
      if (previousGroup.size() != 0) {
        forceComplete(previousGroup);
      } else {
        log.debug("Previous group with id={} has been already released. Skip.", previousGroupId);
      }
    }

    super.handleMessageInternal(message);
    previousGroupId = currentGroupId;   // do unconditionally as we'll check that group on the next step anyway
  }

我意识到这个解决方案使聚合器本身具有状态(虽然不应该如此)并且不适用于许多其他场景。此外,它使应用代码与框架的内部耦合。

如果有人向我指出更好的解决方案,我会很高兴。我愿意提供有关该问题或我所使用的解决方案的更多详细信息。

最佳答案

查看您的 previousGroupId 解决方案和描述:

'look ahead' for a next message to find out if current group is complete or not.

“当前”这个词是这里的关键。

因此,对我来说,这意味着您的聚合器是单个组,您必须释放当前状态并根据当前消息的状态启动一个新状态。

为此目的,我将创建一个真正的单分组聚合器 - correlationStrategy 应该返回一些常量,例如1

是否释放当前组实际上是 ReleaseStrategy 的责任。最新消息实际上是 MessageGroup.getMessages() 中的最后一条消息。因此,您到达该消息,决定是否需要释放,然后继续使用自定义 MessageGroupProcessor 以从聚合器生成累积结果。

MessageGroupProcessor 中,您删除最新的“有罪”消息,不要将其包含到结果中,并将其发送回聚合器以形成新的消息组。

这对你来说有意义吗?

关于java - 如何使用 Spring Integration Aggregator 进行 'look-ahead' 发布?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42562053/

相关文章:

java - [Java][Glpk]求解对偶解

java - 如何用 spring 捕获 hibernate 错误

java - 从不同的 HTTP 请求访问 Spring bean

java - 计划任务响应 channel spring-integration

spring-integration - Spring 集成流程的优雅关闭

JAVA:ThreadGroup.getParent() 上的 SecurityException

java - 屏幕尺寸百分比计算器返回错误尺寸

java - 如何在没有注释的情况下在 Spring 核心中初始化 MongoClient 时注入(inject) Mongo 选项?

java - 如何在 JdbcBatchItemWriter 中使用自动生成的 ID?

spring - 如何使用 Spring Data JPA 根据 bool 参数包含或排除记录?