闲来无事,把之前写的一个游戏服务器框架(《一个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“引发的思考的更多相关文章

  1. Connection reset by peer引发的思考

    http://www.mamicode.com/info-detail-506381.html

  2. Connection reset by peer问题分析

    extremetable导出excel,弹出一个下载窗口,这时不点下载而点取消,则报下面的异常: ClientAbortException Caused by: java.net.SocketExce ...

  3. Connection reset by peer的常见原因

    1,如果一端的Socket被关闭(或主动关闭,或因为异常退出而 引起的关闭),另一端仍发送数据,发送的第一个数据包引发该异常(Connect reset by peer). Socket默认连接60秒 ...

  4. Docker: connection reset by peer

    想来,对docker的学习和实践,已经一年有余了,而我关于这样的文章,只有为数不多的几篇.今天借使用docker中发生的一种异常情况为例,写此篇幅. 这个是在centos7.0 ..netcore2. ...

  5. Connection reset by peer的常见原因及解决办法

    转自:https://blog.csdn.net/xc_zhou/article/details/80950753 1,如果一端的Socket被关闭(或主动关闭,或因为异常退出而 引起的关闭),另一端 ...

  6. 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 ...

  7. 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)"将会导致如下错误:

  8. ab测试出现error: connection reset by peer的解决方案

    我们在使用一些开源程序之前,可能会使用ab工具在服务器或者本地进行一次性能评估,但是很多时候却总是会以失败告终,因为,服务器会拒绝你的ab工具发出的http请求, 出现 error: connecti ...

  9. gem install 出现Errno::ECONNRESET: Connection reset by peer - SSL_connect (https://api.rubygems.org

    在安装了rvm来管理多版本的ruby之后,想在不同环境下安装一些gems,结果gem install puma 之后,发现一次又一次失败. gem install 出现Errno::ECONNRESE ...

随机推荐

  1. 01 选择 Help > Install New Software,在出现的对话框里,点击Add按钮,在对话框的name一栏输入“ADT”,点击Archive...选择离线的ADT文件,contact all update ....千万不要勾选点击Add按钮,在对话框的name一栏输入“ADT”,点击Archive...选择离线的ADT文件,contact all update ....千万不要勾

    引言 好久没碰过android,今天重新搭建了一次环境,遇到的问题记录下载.共以后使用. 安装 软件的软件有jdk+eclipse+adt+sdk 主要记录安装adt和sdk的过程,注意,adt和sd ...

  2. 视频聊天室可以用php制作吗?

    首先,告诉你单纯用php制作视频聊天室是实现不了的,需要配合其他技术手段一起操作,例如和FLASH配合,使用FLASH获取语音(FLASH可以获取访问端的设备,例如摄像头). PHP运行在服务器端,是 ...

  3. iOS 项目改名~~~~~

    今早出现一个改名的囧事...先记录下...在此之前感谢整理过这片文章的作者 --- GarveyCalvin ,插眼!以下是正文---- 前言:在iOS开发中,有时候想改一下项目的名字,都会遇到很多麻 ...

  4. 浅谈 facebook .net sdk 应用

    今天看了一篇非常好的文章,就放在这里与大家分享一下,顺便也给自己留一份.这段时间一直在学习MVC,另外如果大家有什么好的建议或者学习的地方,也请告知一下,谢谢. 这篇主要介绍如何应用facebook ...

  5. 设计模式之美:Manager(管理器)

    索引 意图 结构 参与者 适用性 效果 实现 实现方式(一):Manager 模式的示例实现. 意图 将对一个类的所有对象的管理封装到一个单独的管理器类中. 这使得管理职责的变化独立于类本身,并且管理 ...

  6. 源代码版本管理与项目管理软件的认识与github的注册

    源代码版本管理软件: 主要有:svn,cvs,hg,git,VSS 这些工具主要是一种记录代码更改历史, 可以无限回溯, 用于代码管理,多个程序员开发协作的工具.Perforce,StarTeam)- ...

  7. Form认证的几点说明

    有的页面需要用户认证之后才可以进入,通常都是在Filter的OnActionExecuting方法中我们需要获取当前用户.有两种情况不必登录:1.用户是登录的,也就是认证过的.2.用户上次登录了,但没 ...

  8. FusionCharts简单教程(三)-----FusionCharts的基本属性

          通过前面两章的讲解我们可以制作出简单的图像,但是有时候我们需要对图像进行一个精确的规划,比如设置背景颜色.设置提示信息.设置间隔颜色等等,这时就需要我们对FusionCharts的细节有比 ...

  9. [ACM_贪心] Radar Installation

    http://acm.hust.edu.cn/vjudge/contest/view.action?cid=28415#problem/A 题目大意:X轴为海岸线可放雷达监测目标点,告诉n个目标点和雷 ...

  10. Senparc.Weixin.MP SDK 微信公众平台开发教程(十二):OAuth2.0说明

    紧接上一篇<Senparc.Weixin.MP SDK 微信公众平台开发教程(十一):高级接口说明>,这里专讲OAuth2.0. 理解OAuth2.0 首先我们通过一张图片来了解一下OAu ...