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.rhkb.cn/news/494087.html

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

相关文章

AIGC-------AI生成内容如何赋能AR和VR体验?

AI生成内容如何赋能AR和VR体验 引言 增强现实&#xff08;AR&#xff09;和虚拟现实&#xff08;VR&#xff09;技术近年来蓬勃发展&#xff0c;为用户提供了沉浸式的体验。这些技术已经广泛应用于游戏、教育、医疗、建筑等领域。然而&#xff0c;AR和VR体验的质量与内容的丰富…

VLM--CLIP作分类任务的损失函数

info_nce_loss 这个是clip作对比学习的损失函数 各个博客上都有详细介绍了&#xff0c;我这里就不赘述 def info_nce_loss(image_features, text_features,logit_scale,labels, temperature0.07):batch_size image_features.shape[0]image_features image_features / image…

【模型压缩】原理及实例

在移动智能终端品类越发多样的时代&#xff0c;为了让模型可以顺利部署在算力和存储空间都受限的移动终端&#xff0c;对模型进行压缩尤为重要。模型压缩&#xff08;model compression&#xff09;可以降低神经网络参数量&#xff0c;减少延迟时间&#xff0c;从而实现提高神经…

leetcode-128.最长连续序列-day14

为什么我感觉上述代码时间复杂度接近O(2n), 虽然有while循环&#xff0c;但是前面有个if判断&#xff0c;能进入while循环的也不多&#xff0c;while循环就相当于两个for循环&#xff0c;但不是嵌套类型的&#xff1a; 变量作用域问题&#xff1a;

Burp与其他安全工具联动及代理设置教程

Burp Suite 是一款功能强大的 Web 安全测试工具&#xff0c;其流量拦截和调试功能可以与其他安全工具&#xff08;如 Xray、Yakit、Goby 等&#xff09;实现联动&#xff0c;从而提升渗透测试的效率。本文将详细讲解 Burp 与其他工具联动的原理以及代理设置的操作方法&#xff…

文件操作(File类)

目录 一、初识文件 二、File类 常用方法 一、初识文件 我们目前是如何存储数据的?弊端是什么? int a 1; int[] arr new int[5];我们这些数据是在内存中存储的&#xff0c;是不能够长久保存的。 那么&#xff0c;我们的计算机当中有没有一块硬件可以长久存储数据的? 磁…

Ubuntu硬盘分区及挂载(命令行)

文章目录 一、简介二、硬盘分区三、格式化分区四、自动挂载分区五、调整分区大小小结 一、简介 创建磁盘分区首先需要找出Linux系统中的物理磁盘&#xff0c;在Linux中采用了一种标准格式来为硬盘分配设备名称。 SATA驱动器和SCSI驱动器&#xff1a;设备命名格式为/dev/sdx&a…

用java造1万条数据

上个月项目有造数需求记录一下。 package com.company;public class CreateSqlZhou {public static void main(String[] args) {//insert into Student (id,name,sex,age,adress) values(68881624120312320,zhangsan,男,18,北京);String startSql "insert into Student…

vue iframe进行父子页面通信并切换URL

需求是2个项目需要使用同一个面包屑进行跳转&#xff0c;其中一个是iframe所在的项目&#xff0c;另一个需要通过地址访问。通过 window.parent.postMessage &#xff0c;帮助 <iframe> 内嵌入的子页面和其父页面之间进行跨域通信。 使用通义千问提问后得到一个很好的示…

【Qt】显示类控件:QLabel、QLCDNumber、QProgressBar、QCalendarWidget

目录 QLabel QFrame 例子&#xff1a; textFormat pixmap、scaledContents alignment wordWrap、indent、margin buddy QLCDNumber 例子&#xff1a; QTimer QProgressBar 例子&#xff1a; QCalendarWidget 例子&#xff1a; QLabel 标签控件&#xff0c;用来显示…

UVM 验证方法学之interface学习系列文章(十二)virtual interface 终结篇

一 双向和三态问题 任何具有多个驱动器的信号,都需要使用网(net)来建模。网是唯一能够同时解决不同状态和强度驱动同一信号效果的构造。net的行为由内置解析函数定义,该函数使用net上所有驱动器的值和强度。每当其中一个驱动器发生变化时,就会调用该函数来生成解析值。该…

【游戏设计原理】22 - 石头剪刀布

一、游戏基础&#xff1a;拳头、掌心、分指 首先&#xff0c;石头剪刀布&#xff08;又名“Roshambo”&#xff09;看似简单&#xff0c;实际上可是个“深藏玄机”的零和博弈&#xff08;听起来很高深&#xff0c;其实就是输赢相抵消的意思&#xff09;。游戏中有三种手势&…

iterm2 focus时灰色蒙层出现的解决办法

问题描述&#xff1a; 当前我的iterm2版本是3.5.10&#xff0c;是我最近才更新的&#xff0c;然后就出现以下页面显示问题&#xff0c;如图所示&#xff1a; 我个人对终端、编辑器等使用存在洁癖&#xff0c;尤其是页面显示效果不满意更是不能忍受&#xff0c;之前找了很久没有…

如何在window 使用 conda 环境下载大模型

最近开始学习 变形金刚&#xff0c;最大的问题就是 huggingface 无法访问&#xff0c;无论是翻墙还是通过本地镜像网站HF-Mirror&#xff0c;然后再通过git下载都很慢&#xff0c;影响学习进度&#xff0c;后面看了如下文章&#xff0c;Huggingface配置镜像_huggingface镜像-CS…

Linux 网络维护相关命令简介

目录 零. 概要一. ping二. ip命令2.1 ip address2.2 ip route2.3 ip neighbour 三. traceroute四. DNS查询4.1 nslookup4.2 dig 五. ss 查看网络连接状态 零. 概要 ⏹在Linux系统中有2套用于网络管理的工具集 net-tools 早期网络管理的主要工具集&#xff0c;缺乏对 IPv6、网…

Liveweb视频融合共享平台在果园农场等项目中的视频监控系统搭建方案

一、背景介绍 在我国的大江南北遍布着各种各样的果园&#xff0c;针对这些地处偏僻的果园及农场等环境&#xff0c;较为传统的安全防范方式是建立围墙&#xff0c;但是仅靠围墙仍然无法阻挡不法分子的有意入侵和破坏&#xff0c;因此为了及时发现和处理一些难以察觉的问题&…

Ubuntu vi(vim)编辑器配置一键补全main函数

1.打开对应的配置文件 vi ~/.vim/snippets/c.snippets 2.按G将光标定位到文件末尾 3.按i进入插入模式 以tab键开头插入下的内容&#xff0c;空行也要加 tab键 4.:wq保存退出 5.再打开任意一个新的 .c文件后&#xff0c;插入模式输入 main 然后按tal键就能补全了

javaEE-线程的常用方法-4

目录 一.start():启动一个线程 调用start()方法 start()方法只能调用一次&#xff1a; java中的API: start()和run()的区别: 二.中断一个线程 中断线程方法1:引入标志位 中断线程方法2:调⽤interrupt()⽅法 抛出的异常: 三.等待一个线程 join() 四、获取线程引用 五…

服务器数据恢复—V7000存储中多块磁盘出现故障导致业务中断的数据恢复案例

服务器存储数据恢复环境&#xff1a; 一台V7000存储上共12块SAS机械硬盘&#xff08;其中1块是热备盘&#xff09;&#xff0c;组建了2组Mdisk&#xff0c;创建了一个pool。挂载在小型机上作为逻辑盘使用&#xff0c;小型机上安装的AIXSybase。 服务器存储故障&#xff1a; V7…

2024年图像处理、多媒体技术与机器学习

重要信息 官网&#xff1a;www.ipmml.org 时间&#xff1a;2024年12月27-29日 地点&#xff1a;中国-大理 简介 2024年图像处理、多媒体技术与机器学习&#xff08;CIPMT 2024&#xff09;将于2024年12月27-29日于中国大理召开。将围绕图像处理与多媒体技术、机器学习等在…