系统调用中断(EINTR)与SIGCHLD信号的处理

      • 一、被中断的系统调用(EINTR)的理解
          • 1. 慢系统调用是?
          • 2. 慢系统调用的类别
          • 3. EINTR产生的原因
          • 5. 一般处理方法
      • 二、SIGCHLD信号的处理
          • 1. SIGCHLD信号的产生
          • 2. SIGCHLD信号的处理
          • 3. 不处理SIGCHLD的后果
      • 三、示例代码

一、被中断的系统调用(EINTR)的理解

1. 慢系统调用是?

慢系统调用指可能永远阻塞的系统调用。

也就是处于阻塞状态的系统调用,如果不收到需要的信息,就会一直阻塞在那里。例如accept:在服务器等待客户端建立连接时,如果没有客户端来请求连接,那么accept就会一直阻塞,直到有客户端请求连接为止。像这种系统调用,就称为慢系统调用。

2. 慢系统调用的类别
  • 对管道的读写
  • 对终端设备设备的读写
  • 对网络连接的读写
  • ……

值得注意的是,读写磁盘文件一般不会阻塞,一般会返回给调用者(在没有硬件故障的条件下)

3. EINTR产生的原因

当阻塞于某个慢系统调用的一个进程捕获某个信号且相应信号处理函数返回时,该系统调用可能返回一个EINTR错误。有些内核会自动重启某些被中断的系统调用。

对这句话的理解:
慢系统调用在阻塞状态中,如果收到了某个信号,该系统调用就会被中断,转而去执行对应的信号处理函数,在信号处理函数执行完后,被中断的系统调用就会返回EINTR。有的内核会在被中断的系统调用返回EINTR后重新执行。

accept为例,在服务器阻塞于accept时,这时之前已连接的客户端关闭了,对应该客户端的子进程就会退出,并发送SIGCHLD信号。如果定义了信号处理函数,就会造成accept的中断,然后执行该信号处理函数,执行完后accept就会返回EINTR

5. 一般处理方法

推荐将对accept的调用改为:

while( true ) {int clientlen = sizeof( cliaddr );int connfd = accept( sockfd, (struct sockaddr *)&cliaddr, &clietnlen );if( connfd < 0 ) {if( errnp == EINTR ) {  //如果返回EINTR,就重启该系统调用continue;} else {perror( "accept " );}}// do other things
}

还有两种方法:设置SA_RESTART属性、忽略信号,不过推荐使用上述方法。

二、SIGCHLD信号的处理

1. SIGCHLD信号的产生

在子进程退出时,会发送给父进程一个SIGCHLD信号来表明子进程已退出,此时可以使用wait/waitpid来等待子进程退出并回收子进程占用的资源,避免产生僵尸进程。该信号被默认为忽略。

2. SIGCHLD信号的处理

可以定义该信号的信号处理函数,并使用signal()来运行该信号对应的信号处理函数。

void sig_chld( int signo ) {pid_t pid;int stat;while( ( pid = waitpid( -1, &stat, WNOHANG ) ) > 0 ) {printf("child %d terminated\n", pid );}return ;
}signal( SIGCHLD, sig_chld );

其中,我们必须要为waitpid()函数设置WNOHANG属性,以便告知waitpid在仍有子进程尚未终止时不要阻塞。
循环中不能使用wait()函数,因为我们不能保证wait在还有子进程运行时不会发生阻塞。

在使用循环判断后,就可以将所有已结束的子进程的资源进行回收。

3. 不处理SIGCHLD的后果

如果我们在收到SIGCHLD信号后忽略,而且也没有在父进程最后wait/waitpid子进程结束,就会有僵尸进程产生。僵尸进程虽然不再占有内存等资源,但是会保留进程表中的表项,因此会造成内存泄漏。

linux下,可以使用ps命令查看,stat一栏标志为Z的就是僵尸进程。

三、示例代码

  • server.cpp
#include <iostream>
#include <string.h>
#include <sys/socket.h>
#include <time.h>
#include <unistd.h>
#include <sys/types.h>
#include <fcntl.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <signal.h>
#include <errno.h>
#include <wait.h>
using namespace std;const int PORT = 8080;/* 接收客户端信息 */
void str_echo( int conn_fd ) {ssize_t n;char buf[1024] = {0};while( ( n = read( conn_fd, buf, 1024 ) ) > 0 ) {write( conn_fd, buf, n );}if( n < 0 && errno == EINTR ) {continue;} else if( n < 0 ) {perror( "read " );}
}void sig_chld( int signo ) {pid_t pid;int stat;while( ( pid = waitpid( -1, &stat, WNOHANG ) ) > 0 ) {printf("child %d terminated\n", pid );}return ;
}int main() {int sock_fd = socket( AF_INET, SOCK_STREAM, 0 );if( sock_fd < 0 ) {perror( "socket " );}struct sockaddr_in servaddr;memset( &servaddr, 0, sizeof( servaddr ) );servaddr.sin_family = AF_INET;servaddr.sin_port = htons( PORT );servaddr.sin_addr.s_addr = htonl( INADDR_ANY );int ret = bind( sock_fd, ( struct sockaddr * )&servaddr, sizeof( servaddr ) );if( ret < 0 ) {perror( "bind " );}ret = listen( sock_fd, 5 );if( ret < 0 ) {perror( "bind " );}signal( SIGCHLD, sig_chld );while( true ) {struct sockaddr_in clieaddr;socklen_t len;memset( &clieaddr, 0, sizeof( clieaddr ) );int conn_fd = accept( sock_fd, ( sockaddr * )&clieaddr, &len );if( conn_fd < 0 ) {if( errno == EINTR ) {cout << "EINTR\n";continue;} else {perror( "accept " );}}if( fork() == 0 ) {close( sock_fd );str_echo( conn_fd );exit(0);}close( conn_fd );}return 0;
}
  • client.cpp
#include <iostream>
#include <string.h>
#include <sys/socket.h>
#include <time.h>
#include <unistd.h>
#include <sys/types.h>
#include <fcntl.h>
#include <netinet/in.h>
#include <arpa/inet.h>
using namespace std;const int PORT = 8080;/* 给服务器发送、接收消息的函数 */
void str_cli( FILE *fp, int sock_fd ) {char recvline[1024] = {0}, sendline[1024] = {0};while( fgets( sendline, 1024, fp ) != NULL ) {write( sock_fd, sendline, strlen(sendline) );if( read( sock_fd, recvline, 1024 ) == 0 ) {perror( "readline " );}fputs( recvline, stdout );}
}int main( int argc, char **argv ) {if( argc != 2 ) {cout << "Usage : ./client + ip\n";return 0;}int sock_fd[5] = {0};for( int i = 0; i < 5; i++ ) {  // 创建5个连接sock_fd[i] = socket( AF_INET, SOCK_STREAM, 0 );if( sock_fd[i] < 0 ) {perror( "socket " );}sockaddr_in servaddr;memset( &servaddr, 0, sizeof( servaddr ) );servaddr.sin_family = AF_INET;servaddr.sin_port = htons( PORT );inet_pton( AF_INET, argv[1], &servaddr.sin_addr );int ret = connect( sock_fd[i], (sockaddr *)&servaddr, sizeof( servaddr ) );if( ret < 0 ) {perror( "connect " );}}str_cli( stdin, sock_fd[0] );return 0;
}

运行结果:
1

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/381858.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SMO写的查看数据库信息的代码

要分析一个比较大的数据库&#xff0c;里面的表太多了&#xff0c;虽然是中文命名&#xff0c;但在2005的Management Studio中查看还是比较麻烦&#xff0c;比如&#xff0c;我想查看具有相同字段名称的表的情况就不好办。于是用SMO写了这个东东。代码比较乱&#xff0c;没有进…

win10宽带连接断网自动重连

文章目录1. 断开网络连接&#xff0c;重命名网络连接2. bat代码&#xff1a;检测到断线自动重连3. 设置开机自动执行3.1 方式一&#xff1a;任务计划程序3.2 方式二&#xff1a;用vbs代码开机运行bat1. 断开网络连接&#xff0c;重命名网络连接 中文名字可能出现远程访问错误 …

ubuntu20环境下使用DevStack安装Openstack-Wallaby(单节点、多节点)

文章目录一、单节点部署1. 环境准备1.1 镜像源1.2 pip源1.3 安装依赖包2. OpenStack安装 - wallaby2.1 添加stack用户2.2 设置代理2.3 下载devstack&#xff0c;使用-b指定版本2.4 进入devstack目录&#xff0c;编辑配置文件2.5 开始安装2.6 安装完成二、多节点部署1. 环境准备…

EMF的一些总结(2)——关于EMF的序列化

关于EMF的序列化对于EMF的序列化有几个比较重要的接口&#xff1a;Resource,ResourceSet,Resource.Factory,URIConverter。这些接口的主要作用就是保存模型到持久化存储介质&#xff0c;或者从持久化存储介质加载模型。1&#xff0e;关于URI&#xff08;Uniform Resource Ident…

单链表实例之学生系统

单链表实例之学生系统 #include<stdio.h> #include<stdlib.h>#define NAME_LEN 20typedef struct node {int number;char *name;struct node *next; } node_t;node_t *g_head; int add_stu() {node_t *new, *tmp;tmp g_head;char *name;new (node_t *)malloc(si…

公司培训文档-JavaScript[对象.属性]集锦

SCRIPT 标记 用于包含JavaScript代码. 属性 LANGUAGE 定义脚本语言 SRC 定义一个URL用以指定以.JS结尾的文件 windows对象 每个HTML文档的顶层对象. 属性 frames[] 子桢数组.每个子桢数组按源文档中定义的顺序存放. feames.length 子桢个数. self 当前窗口. parent …

查询分析器下如何备份数据库

查询分析器下如何备份数据库backupdatabasemsdb todiskd:\msdb.bakOK搞定转载于:https://www.cnblogs.com/CnKker/archive/2006/09/26/515318.html

Inside Dynamics Axapta源代码赏析(四)

第八章:Developing Applications Using Business Connector 这一章的代码主要演示如何通过Business Connector与Axapta交互在Dynamics Axapta的客户端安装目录中找到Microsoft.Dynamics.BusinessConnectorNet.dll这个文件,添加到VS.NET的工程中.1.HelloWorldBC.csclassHelloWor…

信号之函数的可重入性

信号之函数的可重入性 在调用某个函数过程中出现中断信号&#xff0c;且改信号处理函数中再次调用该函数&#xff0c;访问全局、静态变量的函数是不可重入函数。 前后数据不一致&#xff0c;函数是不可重入的&#xff0c;特点&#xff1a;函数中使用全局变量或静态变量。 前后数…

ASP.NET 中处理页面“回退”的方法

我们在编写基于 ASP.NET 的应用程序时&#xff0c;如果代码执行出错或检测到异常&#xff0c;一般会提示用户“返回”或“回退”&#xff0c;或者在多步操作、列表/详细的查看界面中&#xff0c;也会给用户提供回退到上一页面的链接&#xff0c;对于这种情况&#xff0c;大家很…

编辑器

1.FCKeditor 编辑器 FCKeditor is compatible with most internet browsers which include: IE 5.5 (Windows), Firefox 1.0, Mozilla 1.3 and Netscape 7. 最新版本:FCKeditor 2.0 语言环境:多国语言特性功能:所见所得,支持平台众多,支持XHTML 1.0,文本格式设置,常见的编辑,复…

VSTO---excel编程 [待续] [12月5日更新,详见文档下面]

最近比较闲了&#xff0c;考试也快要到了。但是编程技术方面还是不能掉啊.现在开始VSTO编程之旅了。这个话题是连Blog的。所以有兴趣的Blog之友&#xff0c;技术之士&#xff0c;可以匿名或者非匿名的评论&#xff0c;交流。对了&#xff0c;下面的代码都是本人写的&#xff0c…

Rose与PowerDesigner:两款建模工具对比分析比较

一、 二者的出身 作为世界最著名的两大CASE工具&#xff0c;Rational Rose和PowerDesigner的名声可谓如雷贯耳。Rose是当时全球最大的CASE工具提供商Rational的拳头产品&#xff0c;UML建模语言就是由Rational公司的三位巨头Booch、Rumbaugh和Jacobson发明的&#xff0c;后来R…

c#活动目录操作

添加引用 System.DirectoryServices导入命名空间 using System.DirectoryServices;srvip "192.168.1.1"; dn "DCl,DCcom";user "administrator"; pwd "123"; DirectoryEntry de;denewDirectoryEntry("LDAP://"srvip &quo…

[导入]Ajax使用初步

文章来源:http://blog.csdn.net/21aspnet/archive/2007/03/19/1534299.aspx 转载于:https://www.cnblogs.com/zhaoxiaoyang2/archive/2007/03/20/816309.html

CodeForces 570B,C

CodeForces 570B 题意&#xff1a;给定n和m&#xff0c;然后再&#xff08;1-n&#xff09;中随机取出c&#xff0c;求一个m使得 的概率最大&#xff0c;概率一样时输出最小的m。 思路&#xff1a;只需要看1到m-1和m1和n的最大的那一边就可以了&#xff0c;坑是n1的情况和n为…

验证码(转)

把下面代码存为一个文件code.aspx。另一个文件里调用<img src"code.aspx">using System;using System.Data;using System.Configuration;using System.Collections;using System.Web;using System.Web.Security;using System.Web.UI;using System.Web.UI.WebCo…

ASP.NET 2.0 的数据源、数据绑定控件概述与区别

一、Render UI 1 GridView GridView 控件用来在表中显示数据源的值。每列表示一个字段&#xff0c;而每行表示一条记录。GridView 控件支持下面的功能&#xff1a; 绑定至数据源控件&#xff0c;如 SqlDataSource。 内置排序功能。 内置更新和删除功能。 内置分页功能。 内…

[导入]javascript总结

1.动态添加一行&#xff0c;和删除当前行<script> var count0; function ff() { var txt1document.getElementById("Text1"); var table1document.getElementById("table1"); rowNotable1.rows.length; Trtable1.insertRow(rowNo); Tr.id"tr&qu…

求助:DataGrid加行号的问题

我的数据是fname,lnameprivatevoidPage_Load(objectsender, System.EventArgs e) { if(!IsPostBack) { myConnectionnew SqlConnection("server127.0.0.1;uidsa;pwdsa;databaseqqq;"); strSQL"SELE…