”Connection reset by peer“引发的思考
闲来无事,把之前写的一个游戏服务器框架(《一个java页游服务器框架》),部署到阿里云服务器上,测试运行了下,结果看到后台log中打印出了“Connection reset by peer”。出于好奇疑问就查了一下相关资料,网上说一般有这几种:
①:服务器的并发连接数超过了其承载量,服务器会将其中一些连接Down掉;
②:客户关掉了浏览器,而服务器还在给客户端发送数据;
③:浏览器端按了Stop
但是这几种都可以排除,刚搭建的服务器,就我测试连接了下不可能超过负载,而且我这是tcp长连接与浏览器没半点关系。后来问了一些大神才知道,一个连接如果长时间不用防火墙或者路由器就会给你断开,于是就会打印如上异常。现实中的解决方法就是,客户端定时向服务器发送心跳包。才知道心跳包原来还有这个作用,说来甚是惭愧啊。
说到了心跳包自然又想到了另一个功能。客户端断电等一些列强制连接断开的情况,可以通过心跳包机制,让服务器端及时知道链接断开,从而及时清除这些无法使用的链接尸体。反过来说,如果没有心跳包,那么客户端强制断开,服务器是无法及时捕获到链接断开的事件的。然而在我测试下,强制关闭客户端,服务器端是会抛出异常的(我的这个框架使用了mina,客户端强制断开后,会调用对应的exceptionCaught方法抛出异常,然后又调用sessionClosed关闭连接)。既然服务器会抛出异常,自然我们不用通过心跳包,服务器端也能及时知道客户端连接断开的情况。
怀着强烈的好奇心就做了如下测试,由于java网络编程有bio和nio以及最新的aio,对于AIO,成产环境中我还没有接触到又用这个的,所以暂不做测试,就针对BIO和NIO做测试
(以下代码比较简单,就不做解释了。仅仅用于测试我的疑惑,某些不合理的地方也就不必在意了)
BIO测试程序:
服务器端测试代码:
public class BIOServer { public static void main(String[] args) throws UnsupportedEncodingException {
byte[] datas="hello".getBytes("UTF-8");
try {
ServerSocket ss = new ServerSocket(810);
while(true){
Socket s = ss.accept();
OutputStream os = s.getOutputStream();
os.write(datas);
}
} catch (IOException e) {
e.printStackTrace();
} } }
客户端程序:
public class Client { public static void main(String[] args) {
try {
Socket s = new Socket("localhost", 810);
InputStream is = s.getInputStream();
byte[] buf = new byte[1024];
int len=0;
while((len=is.read(buf))!=-1){
System.out.println(new String(buf,0,len,"utf-8"));
} System.in.read();
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
} }
先运行服务器端,然后运行客户端收到服务器端发来的hello,然后强制关闭客户端,服务器端没有任何反应,这说明BIO下强制关闭客户端,服务器端是没法及时捕获的。当然在连接断开的情况下,服务器端对此连接进行读写,就会抛出异常。
NIO测试程序:
服务器端:
public class NioServer { public static void main(String[] args)throws Exception { Selector selector = Selector.open();
ServerSocketChannel serverChannel = ServerSocketChannel.open();
serverChannel.configureBlocking(false);
serverChannel.socket().bind(new InetSocketAddress(810));
serverChannel.register(selector, SelectionKey.OP_ACCEPT);
while(true){
int l = selector.select();
if(l==0){
continue;
}
Set<SelectionKey> sets = selector.selectedKeys();
for(SelectionKey key :sets){
if(key.isAcceptable()){
ServerSocketChannel tChannel = (ServerSocketChannel) key.channel();
SocketChannel clientChannel = tChannel.accept();
clientChannel.configureBlocking(false);
clientChannel.register(selector, SelectionKey.OP_READ);
}else if(key.isReadable()){
SocketChannel clientChannel = (SocketChannel) key.channel();
ByteBuffer buf = ByteBuffer.allocate(1024);
int len = clientChannel.read(buf);
if(len==-1){
System.out.println("客户端断开了");
}else{
buf.flip();
System.out.println(new String(buf.array(),0,len,"utf-8"));
}
}
sets.remove(key);
}
}
} }
客户端:
public class Client { public static void main(String[] args) {
try {
Socket s = new Socket("localhost", 810);
OutputStream os = s.getOutputStream();
byte[] datas="hello".getBytes("UTF-8");
os.write(datas);
System.in.read();
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
} }
运行服务器端,然后运行客户端,可以看到服务器端打印出客户端发来的“hello”,然后强制关闭客户端,服务器端就回抛出如下异常:
通过以上测试,可以知道:BIO下,客户端强制断开,服务器端是没法及时知道的;NIO下,客户端强制断开,服务器端可以及时知道并抛出异常。有此差异肯定是因为BIO和nio使用了不同的网络模型。NIO有一个selector,我想也就是这个selector的底层对每一个连接会有检查,所以能及时知道客户端连接断开事件。我的有服务器框架就是基于mina,而mina使用的nio,所以对于强制关闭客户端,我的服务器能及时捕获到并抛出异常这个疑惑,也就算彻底明确了。
”Connection reset by peer“引发的思考的更多相关文章
- Connection reset by peer引发的思考
http://www.mamicode.com/info-detail-506381.html
- Connection reset by peer问题分析
extremetable导出excel,弹出一个下载窗口,这时不点下载而点取消,则报下面的异常: ClientAbortException Caused by: java.net.SocketExce ...
- Connection reset by peer的常见原因
1,如果一端的Socket被关闭(或主动关闭,或因为异常退出而 引起的关闭),另一端仍发送数据,发送的第一个数据包引发该异常(Connect reset by peer). Socket默认连接60秒 ...
- Docker: connection reset by peer
想来,对docker的学习和实践,已经一年有余了,而我关于这样的文章,只有为数不多的几篇.今天借使用docker中发生的一种异常情况为例,写此篇幅. 这个是在centos7.0 ..netcore2. ...
- Connection reset by peer的常见原因及解决办法
转自:https://blog.csdn.net/xc_zhou/article/details/80950753 1,如果一端的Socket被关闭(或主动关闭,或因为异常退出而 引起的关闭),另一端 ...
- java.io.IOException: Connection reset by peer at sun.nio.ch.FileDispatcherImpl.read0(Native Method) at sun.nio.ch.SocketDispatcher.read(SocketDispatcher.java:39)
报错: java.io.IOException: Connection reset by peer at sun.nio.ch.FileDispatcherImpl.read0(Native Meth ...
- Error -27780: [GENERAL_MSG_CAT_SSL_ERROR]connect to host "124.202.213.70" failed: [10054] Connection reset by peer [MsgId: MERR-27780]
解决方案一: 备注: 此方案如果请求响应时间太长,勾选"WinInet replay instead of Sockets(Windows only)"将会导致如下错误:
- ab测试出现error: connection reset by peer的解决方案
我们在使用一些开源程序之前,可能会使用ab工具在服务器或者本地进行一次性能评估,但是很多时候却总是会以失败告终,因为,服务器会拒绝你的ab工具发出的http请求, 出现 error: connecti ...
- gem install 出现Errno::ECONNRESET: Connection reset by peer - SSL_connect (https://api.rubygems.org
在安装了rvm来管理多版本的ruby之后,想在不同环境下安装一些gems,结果gem install puma 之后,发现一次又一次失败. gem install 出现Errno::ECONNRESE ...
随机推荐
- 从MySQL5.7.6开始,安装MySQL提示“请键入 NET HELPMSG 3534 以获得更多的帮助”的解决办法
今天安装MySQL提示如下错误: ----------------------------------------------------------------------------------- ...
- System.Data.SqlClient.SqlError: FILESTREAM 功能被禁用”的错误
还原sql2008数据库时遇到"System.Data.SqlClient.SqlError: FILESTREAM 功能被禁用"的错误,在网上搜索解决方案如下: 1.在" ...
- 使用Ivy管理项目中的依赖
Ivy是什么 Ivy是一个跟踪管理项目直接以来关系的工具.Ivy具有良好的灵活性和可配置性,使其可以适应各种不同的依赖管理和构建过程要求:虽然Ivy作为依赖管理工具,其可以与Apache Ant进行紧 ...
- 使用F#来实现哈夫曼编码吧
最近算法课要求实现哈夫曼编码,由于前面的问题都是使用了F#来解决,偶然换成C#也十分古怪,报告也不好看,风格差太多.一开始是打算把C#版本的哈夫曼编码换用F#来写,结果写到一半就觉得日了狗了...毕竟 ...
- [后端人员耍前端系列]AngularJs篇:使用AngularJs打造一个简易权限系统
一.引言 上一篇博文已经向大家介绍了AngularJS核心的一些知识点,在这篇博文将介绍如何把AngularJs应用到实际项目中.本篇博文将使用AngularJS来打造一个简易的权限管理系统.下面不多 ...
- 图解集合4:HashMap
初识HashMap 之前的List,讲了ArrayList.LinkedList,最后讲到了CopyOnWriteArrayList,就前两者而言,反映的是两种思想: (1)ArrayList以数组形 ...
- debian(kali Linux) 安装net Core
debian(kali Linux) 安装net Core curl -sSL https://raw.githubusercontent.com/dotnet/cli/rel/1.0.0-previ ...
- karma作为jQuery单元测试Runner
karma作为angular测试runner出现,如果你使用过karma一定感受到这很不错的javascript测试runner.简单干净的配置文件karma.config.js,以及karma in ...
- java提高篇(十一)-----强制类型转换
在java中强制类型转换分为基本数据类型和引用数据类型两种,这里我们讨论的后者,也就是引用数据类型的强制类型转换. 在Java中由于继承和向上转型,子类可以非常自然地转换成父类,但是父类转换成子类则需 ...
- Linux Shell函数
200 ? "200px" : this.width)!important;} --> 介绍 正文 $? $?是shell变量,表示"最后一次执行命令"的 ...