如何捕获netty中的所有异常

时间:2015-06-23 04:54:14

标签: java exception-handling netty

据我所知,netty通过覆盖方法exceptionCaught()来处理异常。但我想要的是一个可以处理所有异常的处理程序。所以,管道应该是:

InboundExceptionHandler - inboundHandler1 - inboundHandler2 - outboundHandler1 - outboundHandler2 - OutboundExceptionHandler

这意味着我应该在头部和尾部分开的管道中放置2个异常处理程序。但我觉得它看起来很难看。还有什么好主意吗?

1 个答案:

答案 0 :(得分:20)

您可以在管道的顶部/尾部只有一个入站和出站异常处理程序。如果你想捕获所有异常,你可以做这样的事情(我假设这是Netty 4.0):

import io.netty.channel.*;

import java.net.SocketAddress;

public class ExceptionHandler extends ChannelDuplexHandler {

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
        // Uncaught exceptions from inbound handlers will propagate up to this handler 
    }

    @Override
    public void connect(ChannelHandlerContext ctx, SocketAddress remoteAddress, SocketAddress localAddress, ChannelPromise promise) {
        ctx.connect(remoteAddress, localAddress, promise.addListener(new ChannelFutureListener() {
            @Override
            public void operationComplete(ChannelFuture future) {
                if (!future.isSuccess()) {
                    // Handle connect exception here...
                }
            }
        }));
    }

    @Override
    public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) {
        ctx.write(msg, promise.addListener(new ChannelFutureListener() {
            @Override
            public void operationComplete(ChannelFuture future) {
                if (!future.isSuccess()) {
                    // Handle write exception here...
                }
            }
        }));
    }

    // ... override more outbound methods to handle their exceptions as well
}

入站处理程序抛出的任何异常都会向上传播管道并调用此处理程序的exceptionCaught()方法,假设下面的处理程序不使用它们。

对于write()connect()等出站操作,您需要添加ChannelFutureListener来捕获其异常。 exceptionCaught()方法仅针对来自channelRead()channelActive()等入站事件的例外情况进行调用。

使用此处理程序位于管道的“顶部”,我们可以从下面的所有出站处理程序中捕获异常。假设你的一个出站处理程序正在进行一些编码,这会失败并出现异常,这将由我们添加到write()操作的承诺的通道未来侦听器处理。

如果这个异常处理程序像你最初建议的那样安装在管道的“底部”/头部,那么就不会看到它上面的处理程序的异常,因为如果写的话,永远不会调用它的write()方法在以前的处理程序中失败。这就是为什么这个处理程序必须位于顶部。

为了避免混淆管道的顶部/底部,我将如何配置您的示例管道:

pipeline.addLast(outboundHandler2)        // bottom
        .addLast(outboundHandler1)
        .addLast(inboundHandler2)
        .addLast(inboundHandler1)
        .addLast(new ExceptionHandler()); // top