”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 ...
随机推荐
- WM8978和VS1053B的区别
WM8978是DAC和ADC芯片,不具备编解码的功能.不能把MP3格式或者WAV格式的文件解码出来.有播放和录音功能. 录音如果需要保存为MP3或者WAV格式,那么需要进行软件编码. VS1053B是 ...
- 开源库Magicodes.ECharts使用教程
目录 1 概要 2 2 Magicodes.ECharts工作原理 3 2.1 架构说明 3 2.1.1 Axis 4 2.1.2 CommonD ...
- 让.NET xml序列化支持Nullable
.NET的序列化,关于契约类的生成我们都是通过xsd.exe,对于值类型的可空判断是通过声明同名+Specified的bool属性来判断,比如: public class Person { publi ...
- OAuth 2.0 授权原理
出处:http://www.cnblogs.com/neutra/archive/2012/07/26/2609300.html 最近在做第三方接入的,初步定下使用OAuth2协议,花了些时间对OAu ...
- C#中的线程二(Cotrol.BeginInvoke和Control.Invoke)
C#中的线程二(Cotrol.BeginInvoke和Control.Invoke) 原文地址:http://www.cnblogs.com/whssunboy/archive/2007/06/07/ ...
- 为自己搭建一个鹊桥 -- Native Page与Web View之间的JSBridge实现方式
说起JSBridge,大家最熟悉的应该就是微信的WeixinJSBridge,通过它各个公众页面可以调用后台方法和微信进行交互,为用户提供相关功能.我们就来说说UWP下怎么样实现我们自己的JSBrid ...
- httpclient瓶颈
问题现象: 1.系统异常,应用拒绝访问. 2.web容器线程爆满 问题分析: 1.数据库正常 2.日志信息没有异常 问题思考: 1.应用访问量突破顶峰. 2.应用在某处存在瓶颈 发现问题: 需要了解线 ...
- 新手的第一个phonegap Android应用
对PhoneGap开发感兴趣的请加入群 PhoneGap App开发 348192525 手机成为现在软件应用必不可少的一种设备,然而手机平台的不统一造成我们需要为不同手机重写代码,这对一般应用来 ...
- Git Day02,工作区,暂存区,回退,删除文件
1st,工作区回退:2st,暂存区回退:3rd,删除文件:
- java加密-解密小结
加密算法可以分为 双向加密(对称加密.不对称加密) 单向加密(不可逆加密)—— MD5.sha.hmac... 在对称加密算法中,使用的密钥只有一个,发收信双方都使用这个密钥对数据进行加密和解密 有: ...