recv函数说明返回值

客户端的程序连接上服务器后recv函数阻塞接受,有时会返回0,说明接收超时服务器主动断开了连接,需要重新connect服务器,但重新connect时会报“Transport endpoint is already connected”!!!返回0时正确处理方法是什么呢,大虾指教啊!!!!!

现象:说明服务器主动断开了客户端的连接 客户端应该调用close关闭,然后再连接 原因: 服务器主动关闭,会进入WAIT_TIME状态,需要等待2MSL的时间,导致客户端重连出现“Transport endpoint is already connected” 也可以在服务器端设置端口重用 setsockopt SO_REUSEADDR

socket API 函数 recv经常返回 0,查了一下MSDN,说是优雅的中断。 请问这种错误如何避免。是否要在 recv之前,判定连接是否中断,如果未中断则recv. 

恩。我最后查了一下,是因为服务端关闭了套接字,才导致这边recv返回0。 最后观察了,原来问题是这样的:当客户端不与服务端交互数据好长时间之后,服务端程序会自动断开连接, 同时客户端的连接状态变成了 CLOSE_WAIT.(我用NETSTAT-NA命令查看)。 变成了CLOSE_WAIT之后,如果客户端再向服务端发送数据,然后recv服务端的反馈时,就会造成recv返回0。

服务端程序 固然可以 变成长连接,也就是说 当客户端与服务端 交互状态处于空闲时,比如长达5分钟。服务端就会主动关闭连接,这样可以减轻服务端的压力。如果服务端变成长连接的话,估计服务端连接数压力比较大。

recv函数

int recv( SOCKET s, char FAR *buf, int len, int flags);

不论是客户还是服务器应用程序都用recv函数从TCP连接的另一端接收数据。

该函数的第一个参数指定接收端套接字描述符; 

第二个参数指明一个缓冲区,该缓冲区用来存放recv函数接收到的数据; 

第三个参数指明buf的长度; 第四个参数一般置0。

这里只描述同步Socket的recv函数的执行流程。当应用程序调用recv函数时,

(1)recv先等待s的发送缓冲中的数据被协议传送完毕,如果协议在传送s的发送缓冲中的数据时出现网络错误,那么recv函数返回SOCKET_ERROR,

(2)如果s的发送缓冲中没有数据或者数据被协议成功发送完毕后,recv先检查套接字s的接收缓冲区,

如果s接收缓冲区中没有数据或者协议正在接收数 据,那么recv就一直等待,直到协议把数据接收完毕。

当协议把数据接收完毕,recv函数就把s的接收缓冲中的数据copy到buf中

(注意协议接收到的数据可能大于buf的长度,所以 在这种情况下要调用几次recv函数才能把s的接收缓冲中的数据copy完。

recv函数仅仅是copy数据,真正的接收数据是协议来完成的), recv函数返回其实际copy的字节数。

如果recv在copy时出错,那么它返回SOCKET_ERROR;

如果recv函数在等待协议接收数据时网络中断了,那么它返回0。

默认 socket 是阻塞的 解阻塞与非阻塞recv返回值没有区分,都是 <0 出错 =0 连接关闭 >0 接收到数据大小

特别:

返回值<0时并且(errno == EINTR || errno == EWOULDBLOCK || errno == EAGAIN)的情况下认为连接是正常的,继续接收。

只是阻塞模式下recv会阻塞着接收数据,非阻塞模式下如果没有数据会返回,不会阻塞着读,因此需要循环读取)。

返回说明: 

成功执行时,返回接收到的字节数。

另一端已关闭则返回0。

失败返回-1,

errno被设为以下的某个值 

EAGAIN:套接字已标记为非阻塞,而接收操作被阻塞或者接收超时 

EBADF:sock不是有效的描述词 

ECONNREFUSE:远程主机阻绝网络连接 

EFAULT:内存空间访问出错 

EINTR:操作被信号中断 

EINVAL:参数无效 

ENOMEM:内存不足 

ENOTCONN:与面向连接关联的套接字尚未被连接上 

ENOTSOCK:sock索引的不是套接字 当返回值是0时,为正常关闭连接;

思考:

当对侧没有send,即本侧的套接字s的接收缓冲区无数据,返回值是什么(EAGAIN,原因为超时,待测)

http://hi.baidu.com/passionqiangli/item/505d26fb1021c452c8f33717

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏smy

webpack打包速度和性能再次优化

一. 改单dll为双dll ? 因为上图原因,使用CommonsChunkPlugin时,导致其打包出来的vendors.js内的模块ID会因为其他文件引用模块...

7408
来自专栏SDNLAB

【连载-4】数据中心网络虚拟化 配置管理技术

在构建虚拟网络时,管理员需要进行大量的配置工作,例如端口的ip地址和VXLAN配置等等。显然,没有人愿意在系统每次启动时都将繁琐的配置工作重复一遍,所以将配置信...

2815
来自专栏Albert陈凯

HTTP、TCP、UDP:通信协议的规则和区别

TCP、HTTP、UDP:都是通信协议,也就是通信时所遵守的规则,只有双方按照这个规则“说话”,对方才能理解或为之服务。 TCP HTTP UDP三者的...

3088
来自专栏Python爬虫与算法进阶

强大的异步爬虫 with aiohttp

看到现在网络上大多讲的都是requests、scrapy,却没有说到爬虫中的神器:aiohttp

1732
来自专栏Java帮帮-微信公众号-技术文章全总结

Web-第七天 HTTP&amp;Tomcat学习

HTTP协议:超文本传输协议(HTTP,HyperText Transfer Protocol)是互联网上应用最为广泛的一种网络协议。用于定义WEB浏览器与WE...

1905
来自专栏linux驱动个人学习

预处理

预处理有很多,以下选取我实际用过和见过的: #ifdef 电脑程序语句,我们可以用它区隔一些与特定头文件、程序库和其他文件版本有关的代码。 1 #includ...

2733
来自专栏写代码的海盗

hadoop2.2.0安装需要注意的事情

今天在安装hadoop2.2.0时遇到若干问题,解决这些问题有些心得,记录下来以备不时之需。 问题1、master和slave之间不能相互ssh免密码登陆。  ...

3234
来自专栏Spark学习技巧

高性能:MYSQL异步客户端

实时处理领域,当需要使用外部存储数据染色的时候,需要慎重对待,不能让与外部系统之间的交互延迟对流的整个进度取决定性的影响。

4002
来自专栏人人都是极客

Linux下so动态库一些不为人知的秘密

Linux 下有动态库和静态库,动态库以.so为扩展名,静态库以.a为扩展名。二者都使用广泛。本文主要讲动态库方面知识。

1862
来自专栏桥路_大数据

多云服务器kafka环境搭建并接收flume日志数据

4009

扫码关注云+社区

领取腾讯云代金券