乐趣区

关于java:javaioIOException-远程主机强迫关闭了一个现有的连接异常堆栈信息问题处理屏蔽轰隆隆原创

当咱们应用 Socket 开发服务器间互相通信的时候,应该都遇到这个异样,失常状况下,这个是因为客户端和服务器端网络异样或者强制断开所产出的异样,具体如下:

java.io.IOException: 近程主机强制敞开了一个现有的连贯。at sun.nio.ch.SocketDispatcher.read0(Native Method)
    at sun.nio.ch.SocketDispatcher.read(SocketDispatcher.java:43)
    at sun.nio.ch.IOUtil.readIntoNativeBuffer(IOUtil.java:223)
    at sun.nio.ch.IOUtil.read(IOUtil.java:192)
    at sun.nio.ch.SocketChannelImpl.read(SocketChannelImpl.java:380)
    at io.netty.buffer.PooledByteBuf.setBytes(PooledByteBuf.java:253)
    at io.netty.buffer.AbstractByteBuf.writeBytes(AbstractByteBuf.java:1132)
    at io.netty.channel.socket.nio.NioSocketChannel.doReadBytes(NioSocketChannel.java:350)
    at io.netty.channel.nio.AbstractNioByteChannel$NioByteUnsafe.read(AbstractNioByteChannel.java:151)
    at io.netty.channel.nio.NioEventLoop.processSelectedKey(NioEventLoop.java:719)
    at io.netty.channel.nio.NioEventLoop.processSelectedKeysOptimized(NioEventLoop.java:655)
    at io.netty.channel.nio.NioEventLoop.processSelectedKeys(NioEventLoop.java:581)
    at io.netty.channel.nio.NioEventLoop.run(NioEventLoop.java:493)
    at io.netty.util.concurrent.SingleThreadEventExecutor$4.run(SingleThreadEventExecutor.java:989)
    at io.netty.util.internal.ThreadExecutorMap$2.run(ThreadExecutorMap.java:74)
    at io.netty.util.concurrent.FastThreadLocalRunnable.run(FastThreadLocalRunnable.java:30)
    at java.lang.Thread.run(Thread.java:748)

在网上翻了半天没有找到该异样的解决(屏蔽)办法,于是就本人入手,饥寒交迫吧!
通过查看源码,定位到异样为堆栈信息打印地位:

看名字就能分明这个是默认的异样监听类,该类继承了 ExceptionListenerAdapter,而 ExceptionListenerAdapter 又实现了 ExceptionListener 接口,该接口代码如下:

package com.corundumstudio.socketio.listener;

import io.netty.channel.ChannelHandlerContext;
import java.util.List;
import com.corundumstudio.socketio.SocketIOClient;

public interface ExceptionListener {void onEventException(Exception e, List<Object> args, SocketIOClient client);
    void onDisconnectException(Exception e, SocketIOClient client);
    void onConnectException(Exception e, SocketIOClient client);
    void onPingException(Exception e, SocketIOClient client);
    boolean exceptionCaught(ChannelHandlerContext ctx, Throwable e) throws Exception;
}

如果咱们要屏蔽这个异样(堆栈信息),就须要从新实现这个接口,而后覆写 exceptionCaught 办法即可。
新建 MyDefaultExceptionListener 类继承 ExceptionListenerAdapter,间接上代码:

@Slf4j
public class MyDefaultExceptionListener extends ExceptionListenerAdapter {public MyDefaultExceptionListener() {super();
    }

    @Override
    public void onEventException(Exception e, List<Object> args, SocketIOClient client) {log.error(e.getMessage());
    }

    @Override
    public void onDisconnectException(Exception e, SocketIOClient client) {log.error(e.getMessage());
    }

    @Override
    public void onConnectException(Exception e, SocketIOClient client) {log.error(e.getMessage());
    }

    @Override
    public void onPingException(Exception e, SocketIOClient client) {log.error(e.getMessage());
    }

    @Override
    public boolean exceptionCaught(ChannelHandlerContext ctx, Throwable e) {log.error("谬误:" + e.getMessage());
        ctx.close();
        return true;
    }
}

这个的话,如果出现异常,就不会打印堆栈信息了。

接下来,咱们在 Socket 的初始化配置类外面设置一下,示例代码:

com.corundumstudio.socketio.Configuration configuration = new com.corundumstudio.socketio.Configuration();
//// 其它配置项略 ///
configuration.setExceptionListener(new MyDefaultExceptionListener());
//// 其它配置项略 ///
return  new SocketIOServer(configuration);

这里有个坑须要留神一下,不要在 SocketIOServer 中增加监听,不起作用。

至此,异样覆写实现。

注:本文章应用的是 netty-socketio 作为服务端。引入版本为:

        <!--socket 通信应用 -->
        <dependency>
            <groupId>com.corundumstudio.socketio</groupId>
            <artifactId>netty-socketio</artifactId>
            <version>1.7.19</version>
        </dependency>

实现后日志最终打印成果如下,再没有堆栈信息了:

退出移动版