linux系统编程(五)

1、信号

信号是事件发生时对进程的通知机制,针对每个信号都定义了一个唯一的整数,这些整数定义在signal.h中。

常见信号如下:

  • SIGABRT:进程调用abort函数,系统向进程发送此信号,终止进程并产生核心转储文件。
  • SIGBUS:表示出现了某种内存访问错误;
  • SIGCHLD:父进程的某一子进程终止;
  • SIGINT:用户输入终端终端字符(ctrl+c)
  • SIGKILL:必杀信号,程序无法阻塞、忽略或者捕获
  • SIGPIPE:向管道、FIFO或者socket写入信息时,没有相应的阅读进程;
  • SIGQUIT:键盘输入退出字符(ctrl+\)
  • SIGSEGV:程序对内存的引用无效时会产生此信号。

signal系统调用可以用来改变信号处置:

#include <signal.h>
void (*signal(int sig, void (*handler)(int)))(int);

第一个参数表示需要修改的信号,第二个参数handler是修改后的处置函数,返回值是之前的信号处置函数。

我们可以使用kill来发送信号

#include <signal.h>
int kill(pid_t pid, int sig);

参数pid用于标识一个或者多个目标进程:

  • pid大于0,发送信号给指定进程;
  • pid等于0,发送信号给与调用进程同组的所有进程,包括调用进程自身;
  • pid小于-1,向组ID等于该pid绝对值的进程组内下属进程发送信号;
  • pid等于-1,调用进程有权将信号发往的每一个目标进程,出去init和调用进程自身。特权进程发起这一调用,会发送信号给所有进程,这也被称为广播信号;

如果无进程与指定pid匹配,kill调用失败,errno设置为ESRCH。

除了可以使用kill发送信号,我们还可以使用raise发送信号:

#include <signal.h>
int raise(int sig);

raise是对自身发送信号,相当于调用kill(getpid(), sig)。

以下是示例代码:

#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>extern const char *const sys_siglist[];int sigIntCnt = 0;static void sigHandler(int sig) {switch(sig) {case SIGINT:printf("current sigIntCnt:%d\n", sigIntCnt++);if(sigIntCnt == 3) {exit(1);}break;case SIGQUIT:printf("recevie SIGQUIT, exit!\n");exit(1);break;default:printf("receive msg:%s\n", strsignal(sig));break;}
}int main(int argc, char **argv) {signal(SIGINT, sigHandler);signal(SIGQUIT, sigHandler);int n = 0;while(1) {printf("n:%d\n", n++);sleep(1);}return 0;
}

多个信号可以使用一个称之为信号集的数据结构来标识,该数据类型为sigset_t。下面是一组操作信号集的函数:

#include <signal.h>
int sigemptyset(sigset_t *set);
int sigfillset(sigset_t *set);

创建sigset_t变量后,必须要用上面两个函数来初始化信号集,不能使用memset来初始化。

信号集初始化完成后,可以向信号集中添加或删除单个信号:

#include <signal.h>
int sigaddset(sigset_t *set, int sig);
int sigdelset(sigset_t *set, int sig);

可以用sigismember判断信号集中是否包含某个信号:

#include <signal.h>
int sigismember(const sigset_t *set, int sig);

内核会为每一个进程维护一个掩码(一组信号),阻塞其针对该进程的传递。信号掩码属于线程属性,多线程中每个线程都能使用pthread_sigmask来检查或修改信号掩码。

#include <signal.h>
int sigprocmask(int how, const sigset_t *set, sigset_t *oldset);

使用sigprocmask可以修改信号掩码,也可以获取现有掩码。根据参数how可以确定给掩码带来的变化。

  • SIG_BLOCK:将set信号集内的信号添加到信号掩码中,做并集
  • SIG_UNBLOCK:将set中的信号从当前掩码中移除
  • SIG_SETMASK:将set信号集赋给信号掩码,替换

除了可以用signal来改变信号处置外,还可以使用sigaction做信号处置。

#include <signal.h>
int sigaction(int sig, const struct sigaction *act, struct sigaction *oldact);struct sigaction {void (*sa_handler)(int);sigset_t sa_mask;int sa_flags;void (*sa_restorer)(void);
}

sa_mask定义了一组信号,在调用sa_handler所定义的处理器程序时将阻塞该信号。比如程序运行信号处理函数时,该信号再次到来,信号将不会中断自己。

pause会暂停进程的执行,知道信号处理函数中断该调用。

#include <unistd.h>
int pause();

信号处理函数有一种常见设计:

  • 信号处理函数设置全局性表示变量并退出,主程序对该标志进行周期性检查,发现置位就采取相应动作。进行这种周期性检查时可以让信号处理函数向一个专用管道写入一个字节数据。

我们在设计信号处理函数时要确保处理函数本身是可重入的。信号处理函数可能会更新errno,所以一般情况下进入信号处理函数时记录errno,退出时恢复errno。

如果想要让主程序和信号处理函数共享全局变量,可以进行如下声明:

volatile sig_atomic_t flag;

以下是一个简单的示例:

#include <unistd.h>
#include <signal.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <stdlib.h>static void sigHandler(int sig) {printf("enter sigHandler, sig:%d\n", sig);int cnt = 3;switch(sig) {case SIGINT:while(cnt--) {printf("process SIGINT, current sleep cnt:%d\n", cnt);sleep(1);}break;case SIGQUIT:printf("process SIGQUIT\n");exit(1);break;default:break;}
}int main(int argc, char **argv) {struct sigaction sigact;sigemptyset(&sigact.sa_mask);sigaddset(&sigact.sa_mask, SIGINT);sigaddset(&sigact.sa_mask, SIGQUIT);sigact.sa_handler = sigHandler;sigaction(SIGINT, &sigact, NULL);signal(SIGQUIT, sigHandler);while(1) {sleep(1);}return 0;
}

2、进程

我们可以使用系统调用fork创建一个新的进程,子进程创建时会拷贝父进程的文本段、数据段、堆、栈,但是后续可以各自修改栈、堆中的数据,不影响另一个进程。进程创建完成后,两个进程都会从fork返回处继续执行。

#include <unistd.h>
pid_t fork();

无法创建子进程,fork返回-1。在父进程中fork返回新创建的子进程ID,在子进程中fork返回0。

fork时子进程会得到父进程文件描述符的副本,包含偏移量文件状态标志等。如果子进程更新了文件偏移量,那么也会影响到父进程中的文件描述符。

进程有两种中止方式,一种是异常(abnormal)中止,还有一种是_exit系统调用正常(normal)中止。

#include <unistd.h>
void _exit(int status);

_exit的参数status定义了中止的状态,父进程可以调用wait获取该状态。一般来说状态为0表示进程功成身退,非0值表示进程异常退出。

一般来说程序会使用库函数exit:

#include <stdlib.h>
void exit(int status);

exit会调用退出处理函数、刷新stdio缓冲区、调用_exit。

上面提到了wait,系统调用wait用于等待调用进程的任一子进程中止:

#include <sys/wait.h>
pid_t wait(int *status);

status返回的是子进程的中止状态。

wait调用之后,如果之前已经有子进程中止,wait会立即返回。如果没有子进程中止,调用会一直阻塞直至某个子进程中止。wait返回值为中止子进程的ID,参数status为子进程的返回状态。

如果进程没有子进程,那么wait会返回-1,同时errno会被设置为ECHILD。

以下是一个简单的代码示例:

#include <unistd.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <stdio.h>
#include <sys/wait.h>int main(int argc, char **argv) {int fd = open("tmp.txt", O_RDWR | O_CREAT, 0644);if(fd < 0) {printf("open file failed, err:%s\n", strerror(errno));exit(1);} else {printf("open file success, fd:%d\n", fd);}pid_t pid = fork();if(pid == 0) {printf("Hello, I am child process, id:%d, parent:%d\n", getpid(), getppid());} else if (pid > 0) {printf("Hello, I am parent process, id:%d, child:%d\n", getpid(), pid);} else {printf("fork fail\n");exit(1);}const char *str = pid==0?"child":"parent";off_t offset = lseek(fd, 0, SEEK_CUR);printf("%s pid:%d fd:%d, offset:%ld\n",str, pid, fd, offset);const char *txt = "today is 2024/12/21, now 21:24!";if(pid == 0) {ssize_t written = write(fd, txt, strlen(txt));if(written < strlen(txt)) {printf("%s pid:%d, write fail, written:%ld\n", str, getpid(), written);} else {printf("%s pid:%d, write OK, written:%ld\n", str, getpid(), written);}exit(0);}pid_t child = wait(NULL);if(child > 0) {printf("%s pid:%d, child exit:%d\n", str, getpid(), child);} else {printf("%s pid:%d, child exit fail:%s\n", str, getpid(), strerror(errno));}offset = lseek(fd, 0, SEEK_CUR);printf("%s pid:%d fd:%d, after child write offset:%ld\n",str, pid, fd, offset);close(fd);return 0;
}// 测试结果
/*
open file success, fd:4
Hello, I am parent process, id:8424, child:8425
parent pid:8425 fd:4, offset:0
Hello, I am child process, id:8425, parent:8424
child pid:0 fd:4, offset:0
child pid:8425, write OK, written:31
parent pid:8424, child exit:8425
parent pid:8425 fd:4, after child write offset:31
*/

wait有很多限制:

  • 父进程创建了多个子进程,wait无法等待某个特定子进程完成,只能按顺序等待下一个子进程的中止;
  • 如果子进程退出,wait总是保持阻塞;
  • wait只能发现已经中止的进程,如果子进程因为某个信号而停止,或者收到型号恢复,wait就无能为例了。
#include <sys/wait.h>
pid_t waitpid(pid_t pid, int *status, int options);

参数pid表示要等待的具体子进程:

  • pid等于0,表示等待与调用进程同一进程组的所有子进程;
  • pid小于-1,等待进程标识符与pid绝对值相等的所有子进程;
  • pid等于-1,等待任意子进程,和wait等效。

option是一个掩码,它有以下几个选项:

  • WUNTRACED:除了返回中止子进程的信息外,还会返回因信号而停止的子进程信息;
  • WCONTINUED:因收到SIGCONT信号恢复执行的已经之子进程的状态信息;
  • WNOHANG:如果指定的子进程状态未发生改变,立即返回,不会阻塞。这种情况waitpid返回0。如果没有与指定参数相匹配的子进程,waitpid报错,errno设置为ECHILD。

某一子进程的父进程终止后,它的父进程会变成1(init),这是判断父进程是否存在的方法。

子进程死亡后,内核会将它转换为僵尸进程,父进程需要调用wait来释放子进程资源。如果父进程没有执行wait就退出,init进程会接管子进程并自动调用wait,将僵尸进程从系统移除。

如果父进程创建了许多子进程,但是没有执行wait,那么内核的进程表将永久为该子进程保留一条记录。如果有大量僵尸进程,并且填满了进程表,将会阻碍新进程的创建。这种情况下,只有杀死父进程,才能清理这些僵尸进程。

无论一个子进程什么时候中止,系统都会向父进程发送SIGCHLD信号,默认处理是忽略。我们可以通过安装信号处理程序来捕获它们,用wait来收拾僵尸进程。不过在之前的学习中,用sigaction时我们会设置屏蔽,处理SIGCHLD时如果有新的SIGCHLD到来,此时父进程只会处理一次,这样就可能有漏网之鱼了。所以解决方案是在信号处理函数中添加while:

while(waitpid(-1, NULL, WNOHANG) > 0) {}

以下是一个简单的代码示例:

#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <sys/wait.h>
#include <stdlib.h>static int cnt = 0;
static int total = 0;
static int quit = 0;
static void sigHandler(int sig) {pid_t pid = -1;switch(sig) {case SIGCHLD:while((pid = waitpid(-1, NULL, WNOHANG)) > 0) {printf("recyle pid:%d, cnt:%d\n", pid, cnt++);}if(quit && cnt == total) {printf("recyle last child process, cnt:%d, total:%d, QUIT!\n", cnt, total);exit(0);}break;case SIGINT:case SIGQUIT:if(cnt == total && cnt > 0) {printf("already recyle all child process! cnt:%d, total:%d\n", cnt, total);exit(0);}quit = 1;printf("pid:%d wait for child process stop, cnt:%d, total:%d\n", getpid(), cnt, total);break;default:break;}
}static void childSigHandler(int sig) {printf("pid:%d receive signal:%d, ignore\n", getpid(), sig);
}int main(int argc, char **argv) {struct sigaction act;sigemptyset(&act.sa_mask);sigaddset(&act.sa_mask, SIGCHLD);sigaddset(&act.sa_mask, SIGINT);sigaddset(&act.sa_mask, SIGQUIT);act.sa_handler = sigHandler;sigaction(SIGCHLD, &act, NULL);sigaction(SIGINT, &act, NULL);sigaction(SIGQUIT, &act, NULL);pid_t pid = -1;total = 5;for(int tmp = 0; tmp < total; tmp++) {pid = fork();if(pid == 0) {sigset_t sigset;sigemptyset(&sigset);sigaddset(&sigset, SIGINT);sigaddset(&sigset , SIGQUIT);sigprocmask(SIG_BLOCK, &sigset, NULL);sleep((tmp+1)*2);printf("child process:%d exit!\n", getpid());exit(0);} else {printf("create child %d process OK, pid:%d\n", tmp, pid);}}while(1) {sleep(1);}return 0;
}

3、程序的执行

系统调用execve可以将新程序加载到进程的内存空间,这个过程中将丢弃旧有的程序,进程的栈、数据以及堆会被新程序的相应部件所替换。

#include <unistd.h>
int execve(const char *pathname, char *const argv[], char *const envp[]);

参数pathname包含准备载入当前进程空间的新程序的路径名,可以是绝对路径也可以是相对于当前工作目录的相对路径。参数argv执行了传递给新进程的命令行参数。最后一个参数envp制定了新程序的环境列表,它是一个数组,元素格式为name=value。

execve成功调用后将永不返回,无需检查它的返回值,因为永远为-1。

还有其他的一些系统调用,它们都基于execve:

#include <unistd.h>int execle(const char *pathname, const char *arg, ...);
int execlp(const char *filename, const char *arg, ...);
int execvp(const char *filename, char *const argv[]);
int execv(const char *pathname, char *const argv[]);
int execl(const char *pathname, const char *arg, ...);

execlp和execvp(p:path)允许只提供程序的文件名,系统会在环境变量指定的目录列表中寻找相应的执行文件。

execle、execlp、execl要求开发者在调用中一字符串列表的形式来执行参数,而不是以数组来描述argv列表,首个参数对应于argv[0],必须以NULL指针来中止参数列表,实际填入时需要将NULL转换为char*。

execvp和execv允许开发者用vector(数组)定义参数列表。

execve和execle允许开发者通过envp为新程序显式指定环境变量,envp是以NULL结尾的字符串数组,这些函数以e(environment)结尾。

#include <stdlib.h>
int system(const char *command);

程序可以通过调用system函数来执行任意的shell命令。比如:system(“ls | wc”)。system的优点是简单,代价是低效率。

以下是一个简单的exec使用示例:

#include <stdio.h>int main(int argc, char **argv) {printf("myprint: ");for(int i = 0; i < argc; i++) {printf("%s ", argv[i]);}printf("\n");return 0;
}#include <unistd.h>
#include <sys/wait.h>
#include <stdio.h>int main(int argc, char **argv) {pid_t pid = fork();if(pid == 0) {printf("child process:%d start execle\n", getpid());execle("./myprint", "myprint", "Hello", "World", (char *)NULL, NULL);} else {pid = wait(NULL);printf("parent:%d recyle child:%d OK\n", getpid(), pid);}return 0;
}/*
child process:2920 start execle
myprint: myprint Hello World 
parent:2919 recyle child:2920 OK
*/

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

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

相关文章

Flutter环境搭建

1.Flutter 简介 1.1 Flutter 是什么 &#xff1f; Flutter 是一个 UI SDK&#xff08;Software Development Kit&#xff09;跨平台解决方案&#xff1a;可以实现一套代码发布移动端&#xff08;iOS、Android、HarmonyOS&#xff09;、Web端、桌面端目前很多公司都在用它&…

安全算法基础(一)

安全算法是算法的分支之一&#xff0c;还的依靠大量的数学基础进行计算&#xff0c;本文参照兜哥的AI安全样本对抗&#xff0c;做一个简单的算法安全概括&#xff0c;从零学习。 最新的安全算法对于我们常规的攻击样本检测&#xff0c;效果是不理想的&#xff0c;为了探究其原…

单元测试-Unittest框架实践

文章目录 1.Unittest简介1.1 自动化测试用例编写步骤1.2 相关概念1.3 用例编写规则1.4 断言方法 2.示例2.1 业务代码2.2 编写测试用例2.3 生成报告2.3.1 方法12.3.2 方法2 1.Unittest简介 Unittest是Python自带的单元测试框架&#xff0c;适用于&#xff1a;单元测试、Web自动…

QtCreator配置github copilot实现AI辅助编程

文章目录 1、概述2、配置环境3、演示 1、概述 新时代的浪潮早就已经来临&#xff0c;上不了船的人终将被抛弃&#xff0c;合理使用AI辅助开发、提升效率是大趋势&#xff0c;注意也不要过于依赖。 2024年12月18日&#xff0c;GitHub 官方宣布了一个激动人心的重大消息&#xf…

数字经济下的 AR 眼镜

目录 1. &#x1f4c2; AR 眼镜发展历史 1.1 AR 眼镜相关概念 1.2 市面主流 XR 眼镜 1.3 AR 眼镜大事记 1.4 国内外 XR 眼镜 1.5 国内 AR 眼镜四小龙 2. &#x1f531; 关键技术 2.1 AR 眼镜近眼显示原理 2.2 AR 眼镜关键技术 2.3 AR 眼镜技术难点 3. &#x1f4a…

LabVIEW深海气密采水器测控系统

LabVIEW的深海气密采水器测控系统通过高性价比的硬件选择与自主开发的软件&#xff0c;实现了高精度的温度、盐度和深度测量&#xff0c;并在实际海上试验中得到了有效验证。 项目背景 深海气密采水器是进行海底科学研究的关键工具&#xff0c;用LabVIEW开发了一套测控系统&am…

RocketMQ的集群架构是怎样的?

大家好&#xff0c;我是锋哥。今天分享关于【RocketMQ的集群架构是怎样的?】面试题。希望对大家有帮助&#xff1b; RocketMQ的集群架构是怎样的? 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 RocketMQ 是阿里巴巴开源的分布式消息中间件&#xff0c;广泛用于处…

【Rust自学】4.5. 切片(Slice)

4.5.0. 写在正文之前 这是第四章的最后一篇文章了&#xff0c;在这里也顺便对这章做一个总结&#xff1a; 所有权、借用和切片的概念确保 Rust 程序在编译时的内存安全。 Rust语言让程序员能够以与其他系统编程语言相同的方式控制内存使用情况&#xff0c;但是当数据所有者超…

AI的进阶之路:从机器学习到深度学习的演变(一)

AI的进阶之路&#xff1a;从机器学习到深度学习的演变 在当今科技迅猛发展的时代&#xff0c;人工智能&#xff08;AI&#xff09;、机器学习&#xff08;ML&#xff09;和深度学习&#xff08;DL&#xff09;已成为推动创新的核心力量。这三个领域虽然紧密相连&#xff0c;却…

《算法》题目

多项选择题 2023年2月,美国国家标准与技术研究院(NIST)将 Ascon算法确立为轻量级加密(LWC)标准,关于该算法和标准的说法,正确的是( )。 A.该标准属于国际标准 B.该标准旨在保护物联网(IoT)创建和传输的信息 C.通过法律法规规范标准化机构的职责与权限,可以起到推动技…

Git配置公钥步骤

GIt公钥的配置去除了git push输入账号密码的过程&#xff0c;简化了push流程。 1.生成SSH公钥和私钥 ssh-keygen -t rsa -b 4096 -C “your_emailexample.com” 遇到的所有选项都按回车按默认处理。获得的公钥私钥路径如下&#xff1a; 公钥路径 : ~/.ssh/id_rsa.pub 私钥路径…

Mysql的多表查询及表的连接

Mysql的多表查询及表连接 目录 Mysql的多表查询及表连接连接查询条件有关联的表的连接natural joinusingon等值连接非等值连接 表与表的外连接左外连接右外连接 表的自连接表的子连接表的伪表查询 连接查询条件 查询的两张表如果出现同名的列&#xff0c;我们需要将表名标注到列…

移动魔百盒中的 OpenWrt作为旁路由 安装Tailscale并配置子网路由实现在外面通过家里的局域网ip访问内网设备

移动魔百盒中的 OpenWrt作为旁路由 安装Tailscale并配置子网路由实现在外面通过家里的局域网ip访问内网设备 一、前提条件 确保路由器硬件支持&#xff1a; OpenWrt 路由器需要足够的存储空间和 CPU 性能来运行 Tailscale。确保设备架构支持 Tailscale 二进制文件&#xff0c;例…

每天40分玩转Django:Django部署

Django部署 一、今日学习内容概述 学习模块重要程度主要内容生产环境配置⭐⭐⭐⭐⭐settings配置、环境变量WSGI服务器⭐⭐⭐⭐⭐Gunicorn配置、性能优化Nginx配置⭐⭐⭐⭐反向代理、静态文件安全设置⭐⭐⭐⭐⭐SSL证书、安全选项 二、生产环境配置 2.1 项目结构调整 mypr…

CVPR2024 | 通过集成渐近正态分布学习实现强可迁移对抗攻击

Strong Transferable Adversarial Attacks via Ensembled Asymptotically Normal Distribution Learning 摘要-Abstract引言-Introduction相关工作及前期准备-Related Work and Preliminaries1. 黑盒对抗攻击2. SGD的渐近正态性 提出的方法-Proposed Method随机 BIM 的渐近正态…

华为IPD流程6大阶段370个流程活动详解_第一阶段:概念阶段 — 81个活动

华为IPD流程涵盖了产品从概念到上市的完整过程,各阶段活动明确且相互衔接。在概念启动阶段,产品经理和项目经理分析可行性,PAC评审后成立PDT。概念阶段则包括产品描述、市场定位、投资期望等内容的确定,同时组建PDT核心组并准备项目环境。团队培训涵盖团队建设、流程、业务…

《LangChain大模型应用开发》书籍分享

前言 ChatGPT和OpenAI开发的GPT模型不仅改变了我们的写作和研究方式&#xff0c;还改变了我们处理信息的方式。《LangChain大模型应用开发》讨论了聊天模式下的LLM的运作、能力和局限性&#xff0c;包括ChatGPT和Gemini。书中通过一系列实际例子演示了如何使用LangChain框架构…

Win10将WindowsTerminal设置默认终端并添加到右键(无法使用微软商店)

由于公司内网限制&#xff0c;无法通过微软商店安装 Windows Terminal&#xff0c;本指南提供手动安装和配置新版 Windows Terminal 的步骤&#xff0c;并添加右键菜单快捷方式。 1. 下载新版终端安装包: 访问 Windows Terminal 的 GitHub 发布页面&#xff1a;https://githu…

Oracle中间件 SOA之 OSB 12C服务器环境搭建

环境信息 服务器基本信息 如下表&#xff0c;本次安装总共使用1台服务器&#xff0c;具体信息如下&#xff1a; App1服务器 归类 APP服务器 Ip Address 172.xx.30.xx HostName appdev01. xxxxx.com Alias appdev01 OSB1服务器 归类 OSB服务器 Ip Address 172.xx3…

java小知识点:比较器

java中自主排序主要根据一个Comparator类来实现。 他内部实现用的是Timsort策略。大概思想是说将整个集合分成几个小段&#xff0c;每个小段分别排序&#xff0c;然后再拼在一起。 主要用法是传入两个数&#xff08;也可以不是Integer或int类型&#xff0c;这里只是把他们都统称…