导航
  • 主页
  • 系统维护与使用
  • 应用程序开发
  • 内核源代码
  • 驱动程序开发
  • CPU和硬件区
  • UNIX文化
  • Solaris
  • Power Linux
  • 问答

linux socket如何检测断开

w0911h 2009-07-22 10:30:57
请教在linux下如何快速检测到socket正常或不正常断开,我想通过recv和send的返回值来判断是否连接异常,但是不成功,下面是我的测试代码,连接成功后如果我断开客户端,那么服务端不会打印错误信息而是等待几秒后直接终止;如果断开服务端,客户端的recv不再阻塞,而且返回值一直是0,请高手指点。


//服务端
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <sys/wait.h>
#define MYPORT 3490 /*定义用户连接端口*/
#define BACKLOG 10 /*多少等待连接控制*/
int main(int argc, char *argv[])
{
int sockfd, new_fd;/* listen on sock_fd, new connection on new_fd*/
struct sockaddr_in my_addr; /* my address information */
struct sockaddr_in their_addr; /* connector's address information */
int sin_size;
if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
perror("socket");
exit(1);
}

int opt = 1;
int len = sizeof(opt);
setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &opt, len);

my_addr.sin_family = AF_INET; /* host byte order */
my_addr.sin_port = htons(MYPORT); /* short, network byte order */
my_addr.sin_addr.s_addr = INADDR_ANY; /* auto-fill with my IP */
bzero(&(my_addr.sin_zero),0); /* zero the rest of the struct */

if (bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr))== -1)
{
perror("bind");
exit(1);
}
if (listen(sockfd, BACKLOG) == -1) {
perror("listen");
exit(1);
}

while(1) { /* main accept() loop */
sin_size = sizeof(struct sockaddr_in);
printf("while\n");
if ((new_fd = accept(sockfd, (struct sockaddr *)&their_addr, &sin_size)) == -1)
{
perror("accept");
continue;
}
printf("server: got connection from %s\n", inet_ntoa(their_addr.sin_addr));

while(1)
{
usleep(10000000);
if (send(new_fd, "Hello, world!\n", 14, 0) == -1)
perror("send");
}
close(new_fd);
exit(0);

}
return 0;
}



//客户端
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <sys/wait.h>
#include <netdb.h>
#define PORT 3490 /* 客户机连接远程主机的端口 */
#define MAXDATASIZE 100 /* 每次可以接收的最大字节 */
int main(int argc, char *argv[])
{
int sockfd, numbytes;
char buf[MAXDATASIZE];
struct hostent *he;
struct sockaddr_in their_addr; /* connector's address information */
if (argc != 2)
{
fprintf(stderr,"usage: client hostname\n");
exit(1);
}
if ((he=gethostbyname(argv[1])) == NULL)
{ /* get the host info */
herror("gethostbyname");
exit(1);
}

if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
perror("socket");
exit(1);
}

their_addr.sin_family = AF_INET; /* host byte order */
their_addr.sin_port = htons(PORT); /* short, network byte order */
their_addr.sin_addr = *((struct in_addr *)(he->h_addr));
bzero(&(their_addr.sin_zero),0); /* zero the rest of the struct */
if (connect(sockfd, (struct sockaddr *)&their_addr,sizeof(struct sockaddr)) == -1)
{
perror("connect");
exit(1);
}
while(1)
{
printf("receiving......\n");
if ((numbytes=recv(sockfd, buf, MAXDATASIZE, 0)) == -1)
{
perror("recv");
exit(1);
}
buf[numbytes] = '\0';
printf("Received: %s\n",buf);
}
close(sockfd);
return 0;
}



另外,在网上看到一种利用TCP协议栈中的KeepAlive探测的方法,下面是我从网上找的用的比较多的一段代码,这里应该只是设置了一些属性参数,但不知具体怎样在通信中检测断开,请高手赐教。

#include
……
////KeepAlive实现
//下面代码要求有ACE,如果没有包含ACE,则请把用到的ACE函数改成linux相应的接口
int keepAlive = 1;//设定KeepAlive
int keepIdle = 5;//开始首次KeepAlive探测前的TCP空闭时间
int keepInterval = 5;//两次KeepAlive探测间的时间间隔
int keepCount = 3;//判定断开前的KeepAlive探测次数

if(setsockopt(s,SOL_SOCKET,SO_KEEPALIVE,(void*)&keepAlive,sizeof(keepAlive)) == -1)
{
ACE_DEBUG ((LM_INFO,
ACE_TEXT ("(%P|%t) setsockopt SO_KEEPALIVE error!\n")));

}

if(setsockopt(s,SOL_TCP,TCP_KEEPIDLE,(void *)&keepIdle,sizeof(keepIdle)) == -1)
{
ACE_DEBUG ((LM_INFO,
ACE_TEXT ("(%P|%t) setsockopt TCP_KEEPIDLE error!\n")));
}

if(setsockopt(s,SOL_TCP,TCP_KEEPINTVL,(void *)&keepInterval,sizeof(keepInterval)) == -1)
{
ACE_DEBUG ((LM_INFO,
ACE_TEXT ("(%P|%t) setsockopt TCP_KEEPINTVL error!\n")));
}

if(setsockopt(s,SOL_TCP,TCP_KEEPCNT,(void *)&keepCount,sizeof(keepCount)) == -1)
{
ACE_DEBUG ((LM_INFO,
ACE_TEXT ("(%P|%t)setsockopt TCP_KEEPCNT error!\n")));
}

...全文
2636 点赞 收藏 10
写回复
10 条回复
切换为时间正序
请发表友善的回复…
发表回复
suizouwuya 2012-06-19
通信断开,recv会收到0字节的数据,send系统会发送SIGPIPE信号,可以处理一下这个信号。
回复
iamwjp 2009-08-21
堆栈溢出?
回复
w0911h 2009-07-23
[Quote=引用 2 楼 challenge99 的回复:]
断开客户端和服务器端是什么意思? 说说你具体的操作吧
[/Quote]
在我的测试代码中,服务端一直在send,客户端一直在recv,在连接成功后,如果我直接关闭客户端程序,那么几秒钟之后服务端程序就直接退出了,没有打印出错信息,而且我的循环没有break也不会执行到后面的exit(),为什么服务端就自己退出了,是不是send函数发生错误而导致程序崩溃或是send函数内部调用了exit?
如果我关闭服务端程序,客户端recv会返回0,这时候可以判断为断线

PS,现在帖子怎么不能提前了。。。
回复
海枫 2009-07-22
while(1)
{
printf("receiving......\n");
if ((numbytes=recv(sockfd, buf, MAXDATASIZE, 0)) == -1)
{
perror("recv");
exit(1);
}
buf[numbytes] = '\0';
printf("Received: %s\n",buf);
}

==>
while(1)
{
printf("receiving......\n");
if ((numbytes=recv(sockfd, buf, MAXDATASIZE, 0)) == -1)
{
perror("recv");
exit(1);
}
if(numbytes == 0) break; //对端关闭socket了
buf[numbytes] = '\0';
printf("Received: %s\n",buf);
}
回复
海枫 2009-07-22
>>客户端的recv不再阻塞,而且返回值一直是0

recv函数返回0,表明对端已经断开了,你不应该再在此socket上读数据了!
回复
我想通过recv和send的返回值来判断是否连接异常,但是不成功,

send只要数据写入发送缓冲区就返回了,这样判断没有用的。

除了心跳包,或者类似的办法,没想到有更好的处理方法。

http://topic.csdn.net/t/20050418/11/3945333.html
回复
challenge99 2009-07-22
断开客户端和服务器端是什么意思? 说说你具体的操作吧
回复
kojie_chen 2009-07-22
recv和send不能接收到数据或者不能发送数据,就表示连接已经断开了啊。如果断开了服务器,客户端接收不到数据,当然是一直返回0了
回复
w0911h 2009-07-22
我的项目对实时性要求比较高,数据量不大,但是发送接收很频繁,每秒钟20次以上

[Quote=引用 6 楼 ies_sweet 的回复:]
从TCP来说,
两端中任何一端如果要断开连接,
都应当友好的发送一个shutdown给对方
然后再退出
对端如果收到shutdown自然也会退出,如果收不到那也没有办法

也就是说,没有可靠的办法保证连接两端可以友好可靠的退出

心跳检测是应用上一个不错的办法!
很多实际的业务中都是采用了心跳的方式。(在实时性要求不高的情况下)
[/Quote]
回复
ies_sweet 2009-07-22
从TCP来说,
两端中任何一端如果要断开连接,
都应当友好的发送一个shutdown给对方
然后再退出
对端如果收到shutdown自然也会退出,如果收不到那也没有办法

也就是说,没有可靠的办法保证连接两端可以友好可靠的退出

心跳检测是应用上一个不错的办法!
很多实际的业务中都是采用了心跳的方式。(在实时性要求不高的情况下)
回复
发动态
发帖子
Linux/Unix社区
创建于2007-08-27

2.0w+

社区成员

Linux/Unix社区 应用程序开发区
申请成为版主
社区公告
暂无公告