如何将netty解码器内的消息发送到管道中的下一个处理程序/解码器?

时间:2017-11-15 20:57:32

标签: netty pipeline decoder

我的频道管道包含多个解码器,所有解码器都在TextWebSocketFrame消息上运行。现在我的问题是,我必须根据消息的某些内容选择正确的解码器。

基本上,我必须解析消息中的某个字段,然后决定是否要继续处理消息或将消息传递给下一个编码器/处理程序。

大多数人建议在这种情况下使用单个解码器解码所有消息,但我的问题是一些解码器是动态添加的,将所有逻辑放在单个解码器中会很麻烦。

目前代码如下:

@Override
protected void decode(ChannelHandlerContext ctx, TextWebSocketFrame msg, List<Object> out) throws Exception {
    String messageAsJson = msg.text();
    JsonObject jsonObject = JSON_PARSER.fromJson(messageAsJson, JsonObject.class);

    JsonObject messageHeader = jsonObject.getAsJsonObject(MESSAGE_HEADER_FIELD);
    String serviceAsString = messageHeader.get(MESSAGE_SERVICE_FIELD).getAsString();
    String inboundTypeAsString = messageHeader.get(MESSAGE_TYPE_FIELD).getAsString();

    Service service = JSON_PARSER.fromJson(serviceAsString, Service.class);
    InboundType inboundType = JSON_PARSER.fromJson(inboundTypeAsString, InboundType.class);

    if (service == Service.STREAMING) {
        out.add(decodeQuotesMessage(inboundType, messageAsJson));
    } else {

    }
}

所以基本上我在else分支中需要一些逻辑来将消息传递给管道中的下一个处理程序。

我知道,这种方法不是最有效的方法,但我的服务架构的路径很慢(在不同的线程池上运行),包括这种逻辑和快速路径。因此,我可以在这个地方接受一些慢代码。

1 个答案:

答案 0 :(得分:0)

一般来说,你需要这样的东西:

if (service == Service.STREAMING) {
    ctx.pipeline().addLast(new StreamingHandler());
} else {
    ctx.pipeline().addLast(new OtherHandler());
}
out.add(decodeQuotesMessage(inboundType, messageAsJson));
ctx.pipeline().remove(this);

背后的逻辑是:

  1. 您解码了标题,现在您知道需要遵循的流程;
  2. 根据标题向管道添加特定处理程序;
  3. 您将已解码的消息添加到&#39; out&#39;列表,因此你说&#34;将这个解码的消息发送到管道中的下一个处理程序,这将是步骤2中定义的处理程序,以防当前处理程序在管道中的最后一个&#34 ;;
  4. 您将从管道中删除当前处理程序,以避免处理程序重复,以防您的协议一次又一次地发送相同的标头。但是,这是步骤特定于您的协议,可能没有必要。
  5. 这只是一般方法,但它实际上取决于您的协议流程。