- 1、本文档共6页,可阅读全部内容。
- 2、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
- 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
查看更多
Linuxsocket通信出现CLOSEWAIT
Linux socket通信出现CLOSE_WAIT状态的原因与解决方法
这个问题之前没有怎么留意过,是最近在面试过程中遇到的一个问题,面了两家公司,两家公司竟然都面到到了这个问题,不得不使我开始关注这个问题。说起CLOSE_WAIT状态,如果不知道的话,还是先瞧一下TCP的状态转移图吧。
关 闭socket分为主动关闭(Active closure)和被动关闭(Passive closure)两种情况。前者是指有本地主机主动发起的关闭;而后者则是指本地主机检测到远程主机发起关闭之后,作出回应,从而关闭整个连接。将关闭部 分的状态转移摘出来,就得到了下图:
产生原因通过TCP状态转换图,我们来分析,什么情况下,连接处于CLOSE_WAIT状态呢?在被动关闭连接情况下,在已经接收到FIN,但是还没有发送自己的FIN的时刻,连接处于CLOSE_WAIT状态。通常来讲,CLOSE_WAIT状态的持续时间应该很短,正如SYN_RCVD状态。但是在一些特殊情况下,就会出现连接长时间处于CLOSE_WAIT状态的情况。出现大量close_wait的现象,主要原因是某种情况下对方关闭了socket链接,但是我方忙与读或者写,没有关闭连接。代码需要判断socket,一旦读到0,断开连接,read返回负,检查一下errno,如果不是AGAIN,就断开连接。参考资料4中描述,通过发送SYN-FIN报文来达到产生CLOSE_WAIT状态连接,没有进行具体实验。不过个人认为协议栈会丢弃这种非法报文,感兴趣的同学可以测试一下,然后把结果告诉我;-)为了更加清楚的说明这个问题,我们写一个测试程序,注意这个测试程序是有缺陷的。只要我们构造一种情况,使得对方关闭了socket,我们还在read,或者是直接不关闭socket就会构造这样的情况。server.c:
#include #include #include #define MAXLINE 80#define SERV_PORT 8000int main(void){? ? struct sockaddr_in servaddr, cliaddr;? ? socklen_t cliaddr_len;? ? int listenfd, connfd;? ? char buf[MAXLINE];? ? char str[INET_ADDRSTRLEN];? ? int i, n;? ? listenfd = socket(AF_INET, SOCK_STREAM, 0);? ?? ???int opt = 1;? ?? ???setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, opt, sizeof(opt));? ? bzero(servaddr, sizeof(servaddr));? ? servaddr.sin_family = AF_INET;? ? servaddr.sin_addr.s_addr = htonl(INADDR_ANY);? ? servaddr.sin_port = htons(SERV_PORT);? ? ? ? bind(listenfd, (struct sockaddr *)servaddr, sizeof(servaddr));? ? listen(listenfd, 20);? ? printf(Accepting connections ...\n);? ? while (1) {? ?? ???cliaddr_len = sizeof(cliaddr);? ?? ???connfd = accept(listenfd, ? ?? ?? ?? ?? ? (struct sockaddr *)cliaddr, cliaddr_len);? ?? ???//while (1) ? ?? ?? ?? ?? ? {? ?? ?? ?? ?n = read(connfd, buf, MAXLINE);? ?? ?? ?? ?if (n == 0) {? ?? ?? ?? ?? ? printf(the other side has been closed.\n);? ?? ?? ?? ?? ? break;? ?? ?? ?? ?}? ?? ?? ?? ?printf(received from %s at PORT %d\n,? inet_ntop(AF_INET, cliaddr.sin_addr, str, sizeof(str)), ntohs(cliaddr.sin_port));? ? ? ?? ?? ?? ?fo
文档评论(0)