03 Linux编程-进程
1、进程的相关概念
1.1 程序与进程
程序是静态的概念,进程是程序的一次运行活动。
1.2 查看系统中有哪些进程
ps #只显示一小部分进程
ps -aux #会打印当前所有进程
ps -aux|grep init #使用grep筛选出只含有init的进程top #运行显示的进程有点类似windows的任务管理器
1.3 进程标识符
每个进程都有一个非负整数表示的唯一ID, 叫做PID,类似身份证 。PID=0:称为交换进程(swapper),作用是进程调度 ,PID=1:init进程,作用是系统初始化。
在编程中可以调用getpid函数获取自身的进程标识符。使用getppid获取父进程的进程标识符。
1.4 父进程、子进程
进程A创建了进程B,那么A叫做父进程,B叫做子进程,父子进程是相对的概念,可以理解为人类中的父子关系。
1.5 C程序存储空间的分配

栈,自动变量以及每次函数调用时所需保存的信息都存放在此段中。
堆,通常在堆中进行动态存储分配。
2、进程的创建
//使用fork函数创建一个进程
#include <uinstd.h>
//函数原型pid_t fork(void);
fork();
//fork函数调用成功,返回两次
//返回值为0,代表当前进程是子进程
//返回值非负数,代表当前进程为父进程
//调用失败,返回-1//在使用folk函数创建进程之后的程序,父、子进程都会执行
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>int main()
{pid_t pid;pid_t pid2;pid = getpid();printf("before fork: pid = %d\n", pid);//创建一个子进程fork();pid2 = getpid();printf("after fork: pid = %d\n", pid2);if(pid == getpid()){printf("this is father process\n");}else{printf("this is child process, child pid is %d\n", getpid());}printf("my pid is %d, current pro id is:%d\n", pid, getpid());return 0;
}
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>int main()
{pid_t pid;printf("father : id = %d\n", getpid());//创建一个子进程pid = fork();pid2 = getpid();printf("after fork: pid = %d\n", pid2);if(pid > 0){printf("this is father process, pid = %d\n", getpid());}else if(pid == 0){printf("this is child process, child pid is %d\n", getpid());}printf("my pid is %d, current pro id is:%d\n", pid, getpid());return 0;
}
使用folk函数创建子进程,子进程会获得父进程数据空间、堆和栈的副本。由于在fork之后经常跟随着exec,所以现在的很多实现并不执行一个父进程数据段、栈和堆的完全复制。作为替代,使用了写时复制(Copy-On-Write, COW)技术。这些区域由父、子进程共享,而且内核将它们的访问权限改变为只读的。如果父、子进程中的任一个试图修改这些区域,则内核只为修改区域的那块内存制作一个副本,通常是虚拟存储器系统中的一“页”。
2.1 使用folk创建一个子进程的一般目的
(1)一个父进程希望复制自己,使父、子进程同时执行不同的代码段。这在网络服务进程中是常见的——父进程等待客户端的服务请求。当这种请求到达时,父进程调用fork,使子进程处理此请求。父进程则继续等待下一个服务请求到达。
(2)一个进程要执行一个不同的程序。这对shell是常见的情况。在这种情况下,子进程从fork返回后立即调用exec。
2.2 vfolk和folk函数
vfolk直接使用父进程存储空间,不拷贝。
vfork保证子进程先运行,当子进程调用exit退出后,父进程才执行。
3、进程退出
进程的正常退出有以下几种:通过main函数调用return;进程调用标准c库的exit()函数进行退出; 进程通过调用_exit()或者_Exit()函数退出,属于系统调用;可以通过进程的最后一个线程返回;可以通过最后一个线程调用pthread_exit。
进程的异常退出:调用abort;当进程收到某些信号时,如ctrl+C;最后一个线程对取消(cancellation)请求做出响应。
3.1 父进程等待子进程退出
3.1.1 为什么要等待子进程退出
我们创建子进程的目的是为了让其完成某个任务,我们要知道这个子进程是否完成任务了。通过等待等待子进程退出,我们就可以知道这个子进程的任务完成状态,并且通过检查wait和waitpid所返回的终止状态的宏来收集子进程的退出状态,如果子进程的退出状态不被收集,那么这个进程就会编程僵尸进程。
//包含头文件
#include <sys/types.h>
#include <sys/wait.h>//函数原型
//status参数: 是一个整型数指针//非空: 子进程退出状态放在它所指向的地址中。//空: 不关心退出状态pid_t wait(int *status);
pid_t waitpid(pid_t pid, int *status, int options);
int waitid(idtype_t idtype,id_t id, siginfo_t *infop, int options);//如果其所有子进程都还在运行,则阻塞。
//如果一个子进程已终止,正等待父进程获取其终止状态,则取得该子进程的终止状态立即返回。
//如果它没有任何子进程,则立即出错返回。//区别:wait使调用者阻塞, waitpid有一个选项options,可以使调用者不阻塞,但是不阻塞时,该子进程仍然会变成僵尸进程
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>int main()
{pid_t pid;int cnt = 0;pid = fork();if(pid > 0){wait(NULL);while(1){printf("cnt = %d\n", cnt);printf("this is father print, pid = %d\n", getpid());sleep(1);}}else if(pid == 0){while(1){printf("this is child print, pid = %d\n", getpid());sleep(1);cnt++;if(cnt == 3){exit(0);}}}return 0;
}
3.1.2 孤儿进程
父进程如果不等待子进程退出,在子进程之前就结束了自己的“生命”,此时子进程叫做孤儿进程。Linux避免系统存在过多孤儿进程,init进程收留孤儿进程,变成孤儿进程的新的父进程。
4、exec族函数
4.1.1 exec族函数
exec族函数的作用:当我们用fork函数创建新进程后,经常会在新进程中调用exec函数去执行另外一个程序。当进程调用exec函数时,该进程被完全替换为新程序。因为调用exec函数并不创建新进程,所以前后进程的ID并没有改变。
功能:在调用进程内部执行一个可执行文件。可执行文件既可以是二进制文件,也可以是任何Linux下可执行的脚本文件。
函数族:
exec函数族分别是:execl, execlp, execle, execv, execvp, execvpe
//包含头文件
#include <unistd.h>
extern char **environ;//函数原型
//返回值://exec函数族的函数执行成功后不会返回,调用失败时,会设置errno并返回-1,然后从原程序的调用点接着往下执行。//可以通过perror("why");打印出错误原因
//参数说明://path:可执行文件的路径名字//arg:可执行程序所带的参数,第一个参数为可执行文件名字,没有带路径且arg必须以NULL结束//file:如果参数file中包含/,则就将其视为路径名,否则就按 PATH环境变量,在它所指定的各目录中搜寻可执行文件。原文链接:https://blog.csdn.net/u014530704/article/details/73848573
int execl(const char *path, const char *arg, ...);
int execlp(const char *file, const char *arg, ...);
int execle(const char *path, const char *arg,..., char * const envp[]);
int execv(const char *path, char *const argv[]);
int execvp(const char *file, char *const argv[]);
int execvpe(const char *file, char *const argv[],char *const envp[]);
exec族函数非常难以记忆,函数名中的字符会给我们一些帮助:
l : 使用参数列表
p:使用文件名,并从PATH环境进行寻找可执行文件
v:应先构造一个指向各参数的指针数组,然后将该数组的地址作为这些函数的参数。
e:多了envp[]数组,使用新的环境变量代替调用进程的环境变量
带l的一类exac函数(l表示list),包括execl、execlp、execle,要求将新程序的每个命令行参数都说明为 一个单独的参数。这种参数表以空指针结尾。
//文件execl.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
//函数原型:int execl(const char *path, const char *arg, ...);int main(void)
{printf("before execl\n");if(execl("./bin/echoarg","echoarg","abc",NULL) == -1){printf("execl failed!\n"); }printf("after execl\n");return 0;
}
//文件echoarg.c
#include <stdio.h>int main(int argc,char *argv[])
{int i = 0;for(i = 0; i < argc; i++){printf("argv[%d]: %s\n",i,argv[i]); }return 0;
}
带p的一类exac函数,包括execlp、execvp、execvpe,如果参数file中包含/,则就将其视为路径名,否则就按 PATH环境变量,在它所指定的各目录中搜寻可执行文件。举个例子,PATH=/bin:/usr/bin
//文件execlp.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
//函数原型:int execlp(const char *file, const char *arg, ...);
int main(void)
{printf("before execlp****\n");if(execlp("ps","ps","-l",NULL) == -1){printf("execlp failed!\n");}printf("after execlp*****\n");return 0;
}
带v不带l的一类exac函数,包括execv、execvp、execve,应先构造一个指向各参数的指针数组,然后将该数组的地址作为这些函数的参数。如char *arg[]这种形式,且arg最后一个元素必须是NULL,例如char *arg[] = {“ls”,”-l”,NULL}。
//文件execvp.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
//函数原型:int execvp(const char *file, char *const argv[]);int main(void)
{printf("before execlp****\n");char *argv[] = {"ps","-l",NULL};if(execvp("ps",argv) == -1) {printf("execvp failed!\n"); }printf("after execlp*****\n");return 0;
}
带e的一类exac函数,包括execle、execvpe,可以传递一个指向环境字符串指针数组的指针。 参数例如char *env_init[] = {“AA=aa”,”BB=bb”,NULL}; 带e表示该函数取envp[]数组,而不使用当前环境。
//文件execle.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
//函数原型:int execle(const char *path, const char *arg,..., char * const envp[]);char *env_init[] = {"AA=aa","BB=bb",NULL};
int main(void)
{printf("before execle****\n");if(execle("./bin/echoenv","echoenv",NULL,env_init) == -1){printf("execle failed!\n");} printf("after execle*****\n");return 0;
}
//文件echoenv.c
#include <stdio.h>
#include <unistd.h>
extern char** environ;
int main(int argc , char *argv[])
{int i;char **ptr;for(ptr = environ;*ptr != 0; ptr++)printf("%s\n",*ptr);return 0;
}
4.1.2 exec配合fork使用
实现功能:当父进程检测到输入为1时,创建子进程将配置文件的字段值修改掉。
//文件demo_changeData.c
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h>int main()
{pid_t pid;int data = 10;while(1){print("please input a data\n");scanf("%d", &data);if(data == 1){pid = fork();//pid=0,代表此时是子进程的一个程序段if(pid == 0){int fdSrc;char *readBuf = NULL;fdSrc = open("config.txt", O_RDWR);int size = lseek(fdSrc, 0, SEEK_END);lseek(fdSrc, 0, SEEK_SET);readBuf = (char *)malloc(sizeof(char) * size + 8);int n_read = read(fdSrc, readBuf, size);char *p =strstr(readBuf, "LENG=");if(p == NULL){printf("not fount");exit(-1);}p = p + strlen("LENG=");*p = '5';lseek(fdSrc, 0, SEEK_SET);int n_write = write(fdSrc, readBuf, strlen(readBuf));close(fdSrc);exit(0);}}}}else{printf("wait, do nothing\n");}return 0;
}
//文件demo_exec_fork.c
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h>int main()
{pid_t pid;int data = 10;while(1){print("please input a data\n");scanf("%d", &data);if(data == 1){pid = fork();//pid=0,代表此时是子进程的一个程序段if(pid == 0){//调用demo_changeData文件执行execl("./demo_changeData", "demo_changeData", "config.txt", NULL);}}}else{printf("wait, do nothing\n");}return 0;
}
5、system函数
system函数就是系统封装好的exec和fork结合的函数。
#include <stdlib.h>//函数原型
//返回值://如果 system()调用成功则最后会返回执行shell命令后的返回值;//在调用/bin/sh时失败则返回127;//其他失败原因返回-1;//若参数string为空指针(NULL),则返回非零值。
//注意:在编写具有SUID/SGID权限的程序时请勿使用system(),system()会继承环境变量,通过环境变量可能会造成系统安全的问题。int system(const char *command);//源码
int system(const char * cmdstring)
{pid_t pid;int status;if(cmdstring == NULL){return (1);}if((pid = fork())<0){status = -1;}else if(pid == 0){execl("/bin/sh", "sh", "-c", cmdstring, (char *)0);-exit(127); //子进程正常执行则不会执行此语句}else{while(waitpid(pid, &status, 0) < 0){if(errno != EINTER){status = -1;break;}}}return status;
}
6、popen函数
#include <stdio.h>//函数原型
FILE *popen(const char *conmand,const char *type);
int pclose(FILE *stream);
与system函数相比,popen在应用中的好处在于可以获得函数运行的输出结果。
举例,使用popen获取ps指令的运行结果。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>int main(void)
{char ret[1024] = {0};FILE *fp;fp = popen("ps", "r");int n_read = fread(ret, 1, 1024, fp);printf("read ret %d bytes, ret = %s\n", n_read, ret);return 0;
}
相关文章:
03 Linux编程-进程
1、进程的相关概念 1.1 程序与进程 程序是静态的概念,进程是程序的一次运行活动。 1.2 查看系统中有哪些进程 ps #只显示一小部分进程 ps -aux #会打印当前所有进程 ps -aux|grep init #使用grep筛选出只含有init的进程top #运行显示的进程有点类似windows…...
Hbuild-X运行ios基座app
一、说明 ios真机第一次运行的时候需要下载插件,这个都是自动监测,自动下载的,不用多说。ios真机运行是需要签名的,不然就会报以下错误。如何制作免费的签名证书呢,需要借助爱思助手来完成。 二、安装爱思助手 &…...
Node.js基础:从入门到实战
初识 Node.js 与内置模块 (初识) 1、知道什么是node.js 2、知道node.js可以做什么 3、node.js 中js的组成部分 (内置模块) 4、用 fs 模块读写操作文件 5、使用 path 模块处理路径 6、使用http 模块写一个基本的web服务器 初识 N…...
考研408笔记总结~
目录 一.数据结构 二.计算机组成原理 三.操作系统 四.计算机网络 私以为边看视频,边做笔记会更专注些,大家需要自取。欢迎大家和我一起探讨考研的问题,包括不仅限于专业课,数学,英语等等......,想说什么…...
使用在线工具等方式下载推特视频
使用在线工具等方式下载推特视频 使用在线工具 Visit a Twitter video downloader website: Websites like twdown.net, twittervideodownloader.com, and savevideo.me offer services to download Twitter videos.Paste the Twitter video URL into the designated input bo…...
性能优化:几方面考虑
我们可以继续再考虑下关于性能优化,我们还能从哪些方面着手呢? 1. 代码层面: 使用更高效的数据结构和算法。使用缓存避免多次数据库交互减少不必要的计算和内存分配。利用并行和异步编程提高性能。使用性能分析工具定位和优化瓶颈。 2. We…...
学习大数据:论学习Spark的重要性
随着科技的不断发展,大数据已经成为了当今社会的热门话题。大数据技术的出现,为我们提供了处理海量数据的新方法,使得我们能够从这些数据中挖掘出有价值的信息。在众多的大数据处理框架中,Apache Spark无疑是最为出色的一种。本文…...
学习java第七十一天
DI:依赖注入 依赖注入是spring容器中创建对象时给其设置依赖对象的方式,比如给spring一个清单,清单中列出了需要创建B对象以及其他的一些对象(可能包含了B类型中需要依赖对象),此时spring在创建B对象的时候…...
Altium Designer PCB快捷键设置
6)PCB修改快捷键,并自定义工具栏 添加boardlayerset系统命令。 修改系统脚本, 在D:\Program Files\Altium\AD18\System下,找到advpcb.rcs文件,打开。 Tree MNPCB_LayerSets CaptionManage Layer Se&ts Popup Emp…...
玩转Matlab-Simscape(初级)- 08 - 基于Solidworks、Matlab Simulink、COMSOL的协同仿真(案例实战)
** 玩转Matlab-Simscape(初级)- 08 - 基于Solidworks、Matlab Simulink、COMSOL的协同仿真(案例实战) ** 目录 玩转Matlab-Simscape(初级)- 08 - 基于Solidworks、Matlab Simulink、COMSOL的协同仿真&…...
vue嵌套路由
一、嵌套 children配置 1.父类路由 mymusic 2.子类路由 musicson 1.创建MusicSon组件 <template><div><p>从前和后来</p><p>唯一</p><p>运气来的似有若无</p></div> </template><script>export defaul…...
视频降噪算法 hqdn3d 原理分析
视频降噪 视频降噪是一种处理技术,旨在减少视频中的噪声,提高画面质量。噪声可能来自多种源头,包括摄像机的传感器、压缩算法、传输过程中的干扰等。降噪处理对于视频监控、视频会议、电影后期制作以及任何需要高画质输出的应用场景都非常重…...
Ansys Mechanical|屈曲分析技术
屈曲分析的基本概念 当受拉杆件的应力达到屈服极限或强度极限时,将引起塑性变形或断裂。这些是由于强度不足所引起的失效。 在工程中,我们会注意到当细长杆件受压时,表现出与强度失效完全不同的性质。当杆件受压超过某一临界值时࿰…...
【大模型微调】一文掌握7种大模型微调的方法
本篇文章深入分析了大型模型微调的基本理念和多样化技术,细致介绍了LoRA、适配器调整(Adapter Tuning)、前缀调整(Prefix Tuning)等多个微调方法。详细讨论了每一种策略的基本原则、主要优点以及适宜应用场景,使得读者可以依据特定的应用要求和计算资源限…...
MySQL表突然卡死,删、查操作加载不停解决办法
今天遇到了MySQL删表的时候卡死情况。然后通过网上查阅资料和项目组沟通,了解到了有多人同时对同一张表进行了操作。我和另一个同事同时进行了删除操作,然后另两位同时进行了查询操作,然后还有一位同事用dolphin调度,用datax采集数…...
Rust 标准库的结构及其模块路径
在 Rust 中,标准库提供了一组核心功能,以帮助开发者执行常见的编程任务。当使用这些功能时,我们需要通过特定的模块路径来引用它们。下面,我们将详细介绍 Rust 标准库的结构,并提供相应的 use 路径。 Rust 标准库模块…...
003_PyCharm的安装与使用
如果你正在学习PyQt,本系列教程完全可以带你入门直至入土。 所谓从零开始,就是从软件安装、环境配置开始。 不跳过一个细节,不漏掉一行代码,不省略一个例图。 IDE 开始学习一个编程语言,我们肯定是首先得安装好它&…...
事件传递机制
IOS面试题(UIView) ----- 事件传递机制 - 简书 面试题: 在以下场景中,父视图 ParentView 上有三个子视图 ViewA、ViewB 和 ViewC。ViewA 完全位于 ParentView 的范围内,ViewB 有一半在 ParentView 的范围内,而 ViewC 完全位于 Par…...
DE2-115串口通信
目录 一、 内容概要二、 Hello Nios-II2.1 Nios-II编程2.1.1 硬件Ⅰ 搭建环境Ⅱ 编写代码 2.1.2 软件2.1.3 烧录Ⅰ硬件Ⅱ 软件 2.2 verilog编程 三、 心得体会 一、 内容概要 分别用Verilog和Nios软件编程, 实现DE2-115开发板串口输出“Hello Nios-II”字符到笔记本电脑串口助…...
Danfoss丹佛斯S90泵比例放大器
S90R042、S90R055、S90R075、S90R100、S90R130、S90R180、S90R250电气排量控制变量泵比例阀放大器,电气排量控制为高增益控制方式:通过微小变化的输入电流控制信号即可推动伺服阀主阀芯至全开口位置,进而将最大流量的控制油引入到伺服油缸。伺…...
终极AI图像分层指南:3分钟将复杂插画变成可编辑PSD图层
终极AI图像分层指南:3分钟将复杂插画变成可编辑PSD图层 【免费下载链接】layerdivider A tool to divide a single illustration into a layered structure. 项目地址: https://gitcode.com/gh_mirrors/la/layerdivider 你是否曾面对一幅精美的数字插画&…...
破局双系统文件壁垒:WinBtrfs驱动终极应用指南
破局双系统文件壁垒:WinBtrfs驱动终极应用指南 【免费下载链接】btrfs WinBtrfs - an open-source btrfs driver for Windows 项目地址: https://gitcode.com/gh_mirrors/bt/btrfs 在Windows与Linux双系统环境中,用户常常面临跨系统文件访问的难题…...
丁二酸酯PEG氨基叔丁氧羰基,NHBoc-PEG-SA,可与胺基、羟基等基团发生缩合反应
一.名称英文名称:SA-PEG-NHBoc,Succinic Acid-PEG-NHBoc,NHBoc-PEG-SA,NHBoc-PEG-Succinic Acid中文名称:丁二酸酯聚乙二醇氨基叔丁氧羰基,丁二酸酯PEG氨基叔丁氧羰基分子量:1k,2k&a…...
避坑指南:Doris明细模型(Duplicate Key Model)的5个常见错误及优化方案
避坑指南:Doris明细模型(Duplicate Key Model)的5个常见错误及优化方案 在实时数据分析领域,Apache Doris凭借其卓越的性能和易用性赢得了众多企业的青睐。作为Doris中最基础也最常用的数据模型,明细模型(Duplicate Key Model&…...
如何在5分钟内构建你的专业在线演示文稿:PPTist完全指南
如何在5分钟内构建你的专业在线演示文稿:PPTist完全指南 【免费下载链接】PPTist PowerPoint-ist(/pauəpɔintist/), An online presentation application that replicates most of the commonly used features of MS PowerPoint, allowing …...
2026届毕业生推荐的六大AI辅助论文助手解析与推荐
Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 鉴于“降ai”所表达的意思不清晰确切,猜测围绕这一主题或许是在探究关于AI的热度…...
告别原生依赖:用Docker在Jetson Nano上封装海康威视相机SDK与Python推理应用
边缘视觉革命:基于Docker的海康威视相机SDK与Python推理应用容器化实践 在工业检测、智能安防和自动驾驶等边缘计算场景中,海康威视工业相机与NVIDIA Jetson Nano的组合已成为经典配置。然而,当团队需要批量部署数十台设备时,传统…...
ssh远程登录的时候同一个秘钥可以用于多个不同服务器
可以看到:这2台服务器使用了同一个秘钥,现在都可以正常登录:可以看出来第二个云服务器有安全更新没有激活赶快要更新了。...
手把手排查 DeepSpeed CPUAdam 报错:从 AttributeError 到成功编译 Op 的完整日志分析
深度解析DeepSpeed CPUAdam编译报错:从日志分析到精准修复 当你第一次看到AttributeError: DeepSpeedCPUAdam object has no attribute ds_opt_adam这个错误时,可能会感到困惑。这个错误背后隐藏着DeepSpeed框架中CPUAdam优化器与CUDA环境之间复杂的交互…...
告别单调闪烁!用GD32F303的TIMER高级功能玩转PWM:实现S形曲线呼吸灯与多灯同步效果
解锁GD32F303定时器高阶玩法:S形曲线PWM与多灯协同控制艺术 呼吸灯效果在嵌入式设备中早已司空见惯,但大多数实现仍停留在简单的线性渐变阶段。当LED亮度以恒定速率变化时,人眼会感知到明显的"机械感"——就像早期数字音乐缺少模拟…...
