探讨socket引发SIGPIPE信号的问题
我写socket相关的程序也不是一天两天了,在我的记忆中,只要处理好recv(或read)的返回值中<0,==0,>0三种情况,程序便不会有什么问题。但最近在看公司的源代码时,发现代码中直接将SIGPIPE设置为SIG_IGN。而且附上一段注释:往一个已经关闭的socket写入数据会触发SIGPIPE。我心中顿时一惊,我以前从来没这样做过,虽然没出问题,难道只是我运气好?
周末,决定验证一下。
首先,socket的读事件中,返回0表示socket已关闭。如果已经关闭,再往里面写东西肯定会出问题的,肯定不是这么低级的问题。那么有一种情况,即假如服务端的epoll在一次loop中检测该socket正常,然后去处理其他逻辑。恰好这时客户端这时关闭了socket。接着服务端要往socket写数据。那么,这时是否会出现SIGPIPE呢?于是在网上随便拷贝了几份代码来验证(偷懒...)。
服务端代码
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<sys/socket.h>
#include<sys/types.h>
#include<unistd.h>
#include<netinet/in.h>
#include <errno.h>
#define PORT 6666
int main(int argc,char **argv)
{
int ser_sockfd,cli_sockfd;
int err,n; struct sockaddr_in ser_addr;
struct sockaddr_in cli_addr;
char recvline[];
const char *sendline = "hello client"; ser_sockfd=socket(AF_INET,SOCK_STREAM,);
if(ser_sockfd==-)
{
printf("socket error:%s\n",strerror(errno));
return -;
} bzero(&ser_addr,sizeof(ser_addr));
ser_addr.sin_family=AF_INET;
ser_addr.sin_addr.s_addr=htonl(INADDR_ANY);
ser_addr.sin_port=htons(PORT);
err=bind(ser_sockfd,(struct sockaddr *)&ser_addr,sizeof(ser_addr));
if(err==-)
{
printf("bind error:%s\n",strerror(errno));
return -;
} err=listen(ser_sockfd,);
if(err==-)
{
printf("listen error\n");
return -;
} printf("listen the port:%d\n",PORT); while()
{
socklen_t addlen=sizeof(struct sockaddr);
cli_sockfd=accept(ser_sockfd,(struct sockaddr *)&cli_addr,&addlen);
if(cli_sockfd==-)
{
printf("accept error\n");
}
while()
{
printf("waiting for client...\n");
n=recv(cli_sockfd,recvline,,);
printf( "recv return %d\n",n );
if(n==-)
{
printf("recv error\n");
}
recvline[n]='\0'; printf("recv data is:%s\n",recvline); int ret = send(cli_sockfd,sendline,strlen(sendline),);
printf( "send return %d\n",ret );
}
close(cli_sockfd);
} close(ser_sockfd); return ;
}
客户端代码
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<sys/socket.h>
#include<sys/types.h>
#include<unistd.h>
#include<netinet/in.h>
#include <arpa/inet.h> #define PORT 6666 int main(int argc,char **argv)
{
int sockfd;
int err,n;
struct sockaddr_in addr_ser;
const char *sendline = "hello baby"; sockfd=socket(AF_INET,SOCK_STREAM,);
if(sockfd==-)
{
printf("socket error\n");
return -;
} bzero(&addr_ser,sizeof(addr_ser));
addr_ser.sin_family=AF_INET;
inet_aton( "127.0.0.1",&addr_ser.sin_addr );
addr_ser.sin_port=htons(PORT);
err=connect(sockfd,(struct sockaddr *)&addr_ser,sizeof(addr_ser));
if(err==-)
{
printf("connect error\n");
return -;
} printf("connect with server...\n"); send(sockfd,sendline,strlen(sendline),); close( sockfd ); printf("already close...\n"); return ;
}
这个测试很简单,服务器打开端口监听,客户端连接上服务器后马上发送数据,接着直接关闭socket并退出程序。服务器在收到消息后,断点等待客户端退出,然后往socket写数据,看是否会出现SIGPIPE。
服务器编译为sock_s,客户端编译sock_c,先用gdb运行sock_s,下好断点,然后运行sock_c。
Reading symbols from ./sock_s...done.
(gdb) b
Breakpoint at 0x400b7b: file sock_s.cpp, line .
(gdb) r
Starting program: /home/xzc/code/test/sock_s
listen the port:
waiting for client...
recv return
recv data is:hello baby Breakpoint , main (argc=, argv=0x7fffffffded8) at sock_s.cpp:
int ret = send(cli_sockfd,sendline,strlen(sendline),);
(gdb) c
Continuing.
send return
waiting for client...
recv return
recv data is: Breakpoint , main (argc=, argv=0x7fffffffded8) at sock_s.cpp:
int ret = send(cli_sockfd,sendline,strlen(sendline),);
(gdb) c
Continuing. Program received signal SIGPIPE, Broken pipe.
0x00007ffff7b104fd in __libc_send (fd=, buf=0x400c74, n=, flags=-)
at ../sysdeps/unix/sysv/linux/x86_64/send.c:
../sysdeps/unix/sysv/linux/x86_64/send.c: 没有那个文件或目录.
由上面的调试结果可以看出,客户端连接上后发了hello baby然后直接退出。这时服务器在发送数据之前断点停下。我故意等了很久,确认客户端已真正退出,再continue,可以看到,send函数没有异常,返回了12,说明数据正常写到了socket缓冲区。这时socket会再次收到recv,长度为0,表示socket关闭。我故意不处理,继续send,这次gdb捕捉到了SIGPIPE。
在recv返回0之前用netstat查看网络状态:
...
tcp localhost: localhost: FIN_WAIT2
tcp localhost: localhost: CLOSE_WAIT
...
在recv返回0之后,虽然服务器依然没有调用close,连接在网络状态已经查询不到,说明已关闭。但如果在recv返回0之前,断点很久,客户端的socket已关闭,服务端则仍然是CLOSE_WAIT,结果依然如上。
后来又用了epoll模型来测试,结果不变。
因此,recv返回0则认为程序已经知道socket关闭,socket底层将会正常执行TCP的断开流程,跟你是否调用close无关。是否调用close只是是否释放文件描述符等资源。所以,处理好socket的断开事件,则无需理会SIGPIPE.
探讨socket引发SIGPIPE信号的问题的更多相关文章
- SIGPIPE信号详解
转自:http://blog.csdn.net/lmh12506/article/details/8457772 前一段面试的时候被问到项目中有没有处理SIGPIPE信号,怎么处理的?当时没有答出来, ...
- SIGPIPE信号
#include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include &l ...
- 忽略SIGPIPE信号
#include <stdlib.h> #include <sys/signal.h> void SetupSignal() { struct sigaction sa; // ...
- 关于SIGPIPE信号
对一个对端已经关闭的socket调用两次write, 第二次将会生成SIGPIPE信号, 该信号默认结束进程.具体的分析可以结合TCP的"四次握手"关闭. TCP是全双工的信道, ...
- [转]SIGPIPE信号
我写了一个服务器程序,在Linux下测试,然后用C++写了客户端用千万级别数量的短链接进行压力测试. 但是服务器总是莫名退出,没有core文件. 最后问题确定为, 对一个对端已经关闭的socket调 ...
- Linux网络编程-SIGPIPE信号导致的程序退出问题
当客户端close关闭连接时,若server端接着发送数据,根据TCP协议的规定,server端会收到RST响应,当server端再次往客户端发送数据时,系统会发出一个SIGPIPE信号给server ...
- SIGPIPE信号解析
当服务器close一个连接时,若client端接着发数据.根据TCP协议的规定,会收到一个RST响应,client再往这个服务器发送数据时,系统会发出一个SIGPIPE信号给进程,告诉进程这个连接已经 ...
- socket编程时SIGPIPE信号的处理
如果在write调用期间对方关闭连接,视时间顺序的不同有以下几种情况: 1. 刚好在write调用之前对方关闭: write返回失败,同时产生SIGPIPE. 2. write调用过程中对方关闭: 返 ...
- TCP/IP SIGPIPE信号
往一个已经接受FIN的套接中写是允许的,接受到FIN仅仅代表对方不再发送数据. 在收到RST段之后,如果在调用write就 会产生SIGPIPE信息,对于这个信号的处理我们通常 解决方法 signal ...
随机推荐
- 一个菜鸟所喜欢用的响应式布局,操作方便简单、时尚简约,适合新手!(一个Dreamweaver cs6生成响应式布局)
前端开发并不是一个容易的工作,不仅需要掌握HTML.CSS和JavaScript,针对不同的浏览器版本和平台,还需要了解如何设计出跨平台的网站.如今随着响应式设计的流行,前端开发变得越来越困难,且花费 ...
- 将apk文件添加到Android模拟器(AVD)中运行
apk不同exe和jar文件,apk需要在安卓系统中运行,单有一个apk文件还是没用,不能直接拖进AVD中(当然可以直接放到安卓系统的手机中) 由于我们的eclipse大都是已经安装好,解压直接使用的 ...
- WPF 制作圆角按钮
在程序对应坐置插入以下代码,或是先拖一个按钮控件到窗体中,再替换对应的代码. 修改 CornerRadius="18,3,18,3" 就可以改变圆角大小 按钮效果: <Bu ...
- Mindset + Know-how+Concepture + Methodology+Technology
在做成都专案过程中深刻体会到一个IT全才应该具备Domain Know-how,拥有正确地理念是十分重要的, 我想只能是某几个领域,专案管理,专案运行中的各种手法,即Methodology,最后才是K ...
- css float父元素高度塌陷
css float父元素高度塌陷 float 使父元素高度塌陷不是BUG,反而是标准. float 原本是为了解决文字环绕才出现的. 当然有的时候要解决高度塌陷的问题 以下几个方法可以解决float ...
- Sql Server同步之订阅
1.新建一个订阅 2.订阅新建完成之后,先选择发布端 3.选择需要同步的组 4.选择目标数据库 5.选择链接发布端方式,采用sql server login 6.选择执行同步的计划 7.选择是立马执行 ...
- 关于winform主题IrisSkin2的编写
第一步:首先引用IrisSkin2.dll. 第二步自定义类: /// <summary> /// 窗体主题边界类 /// </summary> public class Fo ...
- uva 469 - Wetlands of Florida
题目:给你一个矩阵和某些点,找到给的点所处连续的W区域的面积(八个方向). 分析:搜索.floodfill算法,利用搜索直接求解就可以了. 说明:注意读入数据的格式. #include <cst ...
- H5原生拖拽事件
使用原生js实现简单的拖拽事件 <!DOCTYPE html> <html lang="en"> <head> <meta charset ...
- js拖动层
模仿网易彩票网(http://caipiao.163.com/)的登陆框自己做了一个拖动层,不过有点小问题——在谷歌浏览拖动的时候鼠标状态变成了文字状态(cursor:text;) <!DOCT ...