Linux进程编程(PS: exec族函数、system、popen函数)

目录

  • 1.进程相关概念
    • 程序和进程
    • 查看系统中的进程
      • ps指令
      • top指令
    • 进程标识符 使用getpid()获取
    • 父进程,子进程
  • 2.创建进程fork
    • 进程创建发生了什么——C程序的存储空间如何分配
  • 3.创建进程vfork(区别fork)
  • 4.进程退出
    • 正常退出
    • 异常退出
  • 5.父进程等待子进程退出
    • 父进程收集子进程退出状态(僵尸进程)
    • 等待退出函数wait()
    • 等待退出函数waitpid()
    • 父进程先于子进程退出(孤儿进程)
  • 6.exec族函数(让子进程调用其他程序)
    • execl,execlp
    • execv execvp
    • exec配合fork使用
    • system函数
    • popen函数

1.进程相关概念

程序和进程

程序是静态的概念,gcc xx.x -o pro,磁盘中生成的pro就是程序。

进程是程序的一次运行活动,通俗的讲就是程序跑起来了,系统中就多了一个进程。

查看系统中的进程

ps指令

查看系统中所有进程

ps -aux

结果:
在这里插入图片描述

查看系统中的init进程

ps -aux | grep init

结果:
在这里插入图片描述
即把ps -aux指令所有输出的结果通过管道导向grep进行搜索,查找init关键字的文本

-aux 显示所有包含其他使用者的进程
|管道符号
grep 用于查找文件里符合条件的字符串

top指令

类似windows的任务管理器,数据也是实时动态变化的。
在这里插入图片描述

进程标识符 使用getpid()获取

每一个进程都有一个非负整数标识唯一的ID,即为pid。

调用getpid()获取自身进程标识符,getppid()获取获取父进程标识符

其中系统所占用的进程标识符如下:

pid进程名称作用说明
pid = 0交换进程用于进程调度所有“同时”在运行的程序所占用的资源受到进程调度的影响
pid = 1init进程用于系统初始化程序运行,内核加载完毕,文件系统起来的第一个进程就是init进程,读取配置文件然后再启动其他进程。(如ktv点歌机开机后看到的是点歌界面,而不是字符界面)
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>int main()
{//pid_t getpid(void);pid_t pid;pid=getpid();printf("我的pid是:%d\n",pid);return 0;
}

父进程,子进程

进程A创建了进程B,A就是父进程,B就是子进程

2.创建进程fork

pid_t fork(void);

fork函数调用成功,返回两次:返回值为0,代表当前进程是子进程,非负数为父进程,如果调用失败则返回-1
创建子进程的目的:复制父进程(此时两个或两个以上进程),父进程等待客户端服务请求,当这种请求到达时,父进程调用fork,让子进程去处理(QQ服务器 客户端 结合Socket网络编程)

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>int main()
{pid_t pid;pid =  getpid();fork();//此处开始,后面所有代码执行了两次 //fork()有两个返回值printf("my pid is %d\n",pid);return 0;
}

返回两次:
在这里插入图片描述

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>int main()
{pid_t pid;pid =  getpid();fork();printf("my pid is %d,current pro id:%d\n",pid,getpid());return 0;
}                                    

在这里插入图片描述

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>int main()
{pid_t pid;pid =  getpid();fork();if(pid == getpid()){printf("this is father print\n");}else{printf("this is child print,child pid = %d\n",getpid());}return 0;
}                                                                                                                                                                                                              

在这里插入图片描述

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>int main()
{pid_t pid;printf("father: pid = %d\n",getpid());pid =  fork();if(pid > 0){printf("this is father print, pid = %d\n",getpid());}else if(pid == 0){printf("this is child print,child pid = %d\n",getpid());}return 0;
}

在这里插入图片描述

应用场景(模拟网络等待):

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>int main()
{pid_t pid;int data;while(1){printf("please input a data\n");scanf("%d",&data);if(data == 1){pid = fork();if(pid > 0){}else if(pid == 0){while(1){printf("do net request,pid=%d\n",getpid());sleep(3);}}}else{printf("wait, do nothing\n");}}return 0;
}

进程创建发生了什么——C程序的存储空间如何分配

详细参照博文:内存四区(代码区 静态区 栈区 堆区)

fork创建进程,所有的东西都进行了拷贝,包括全局变量、局部变量、代码等,各进程内改变变量的值互不影响。

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

详细说明:
在这里插入图片描述

3.创建进程vfork(区别fork)

vfork与fork的区别:
1.(共用)vfork直接使用父进程的存储空间,不拷贝(随着内核的发展,fork目前执行的是写时拷贝—copy on write 若子进程未改变原始变量的值时不会进行拷贝)
2.(等待)vfork保证子进程先运行,当子进程调用exit退出后,父进程才执行

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>int main(){int cnt=0;pid_t pid;pid_t repid;pid=getpid();repid=vfork();if(repid > 0){while(1){printf("cnt=%d\n",cnt);printf("this is father pid,repid=%d\n",repid);sleep(1);}}else if(repid == 0){while(1){printf("this is child pid,repid=%d\n",repid);sleep(1);cnt++;if(cnt==2){exit(0);}}}return 0;
}

运行结果:
在这里插入图片描述

4.进程退出

正常退出

1.main函数调用return

2.进程调用exit(),标准C库

3.进程调用_exit()或者_Exit(),系统调用

4.进程最后一个线程返回

5.最后一个线程调用pthread_exit

异常退出

1.调用abort

2.当进程收到某些信号时,如ctrl+c

3.最后一个线程对取消(cancellation)请求作出响应

注意:不管进程如何终止,最后都会执行内核中的同一段代码,这段代码为相应进程关闭所有打开描述符,释放它所使用的存储器等。

5.父进程等待子进程退出

父进程收集子进程退出状态(僵尸进程)

父进程等待子进程退出并收集子进程退出状态
子进程退出状态不被收集,会变成僵尸进程

僵尸进程的例子:

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>int main()
{pid_t pid;int i;int cnt=0;pid=fork();if(pid > 0){//父进程while(1){printf("这是父进程,pid=%d\n",getpid());printf("cnt=%d\n",cnt);sleep(2);//防止刷屏}	}else if(pid == 0){//子进程for(i=0;i<5;i++){//看看是不是保证子进程先运行,五次过后推出进入父进程printf("这是子进程,pid=%d,这是第%d次\n",getpid(),i+1);cnt++;sleep(1);//防止刷屏}exit(-1);}return 0;
}

在这里插入图片描述

此时运行的结果是子进程退出了,但是退出状态没有被收集,子进程成了僵尸进程(zomb)。

在这里插入图片描述

等待退出函数wait()

在这里插入图片描述

wait(int *status):
status参数,他是一个整型数指针。
非空:子进程退出状态放在它所指向的地址
空(NULL):不关心退出状态

在这里插入图片描述

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>int main()
{pid_t pid;int i;int cnt=0;int status;pid=fork();if(pid > 0){//父进程wait(&status);printf("子进程退出,status:%d\n",WEXITSTATUS(status));//WEXITSTATUS()是解析子进程退出码的宏,下面有详细介绍while(1){printf("这是父进程,pid=%d\n",getpid());printf("cnt=%d\n",cnt);sleep(2);//防止刷屏}	}else if(pid == 0){//子进程for(i=0;i<5;i++){//看看是不是保证子进程先运行,五次过后推出进入父进程printf("这是子进程,pid=%d,这是第%d次\n",getpid(),i+1);cnt++;sleep(1);//防止刷屏}exit(3);//退出码返回给父进程中的wait进行收集}return 0;
}

在这里插入图片描述

解析出wait()对子进程退出码回收的宏

在这里插入图片描述

等待退出函数waitpid()

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述
来个例子:

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>int main()
{pid_t pid;int i;int cnt=0;int status;pid=fork();if(pid > 0){//父进程waitpid(pid,&status,WNOHANG);//WNOHANG不阻塞printf("子进程退出,status:%d\n",WEXITSTATUS(status));while(1){printf("这是父进程,pid=%d\n",getpid());printf("cnt=%d\n",cnt);sleep(2);//防止刷屏}}else if(pid == 0){//子进程for(i=0;i<5;i++){//看看是不是保证子进程先运行,五次过后推出进入父进程printf("这是子进程,pid=%d,这是第%d次\n",getpid(),i+1);cnt++;sleep(1);//防止刷屏}exit(3);}return 0;
}

结果如下所示,发现子进程也变为僵尸进程。
在这里插入图片描述

在这里插入图片描述

父进程先于子进程退出(孤儿进程)

父进程如果不等待子进程退出 ,在子进程之前就结束了自己的生命,此时的子进程就叫做是孤儿进程
Linux避免系统存在太多的孤儿进程,init进程收留孤儿进程,变成孤儿进程的父进程

例子:

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>int main()
{pid_t pid;int i;int status;int cunt=0;pid=fork();if(pid > 0){printf("这是父进程,pid=%d\n",getpid());printf("cunt=%d\n",cunt);sleep(2);}else if(pid == 0){for(i=0;i<5;i++){printf("这是子进程,pid=%d,我的父进程的pid=%d\n",getpid(),getppid());cunt++;sleep(2);}exit(3);}return 0;
}

运行结果:
在这里插入图片描述

最后再来一个综合的例子:
在这里插入图片描述

6.exec族函数(让子进程调用其他程序)

参照博文:https://blog.csdn.net/u014530704/article/details/73848573

execl,execlp

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>int main(){//int execl(const char *path, const char *arg, .../* (char  *) NULL */);
/*      path:可执行文件的路径名字arg:可执行程序所带的参数,第一个参数为可执行文件名字,没有带路径且arg必须以NULL结束file:如果参数file中包含/,则就将其视为路径名,否则就按 PATH环境变量,在它所指定的各目录中搜寻可执行文件exec族函数参数极难记忆和分辨,函数名中的字符会给我们一些帮助:l : 使用参数列表p:使用文件名,并从PATH环境进行寻找可执行文件v:应先构造一个指向各参数的指针数组,然后将该数组的地址作为这些函数的参数。e:多了envp[]数组,使用新的环境变量代替调用进程的环境变量exec函数族的函数执行成功后不会返回,调用失败时,会设置errno并返回-1,然后从原程序的调用点接着往下执行。 
*/printf("before execl\n");if(execl("/bin/ls","ls",NULL,NULL) == -1)//通过whereis指令找到ls命令的位置{printf("execl failed!\n");perror("why");}printf("after execl\n");return 0;
}
int main(){//int execlp(const char *file, const char *arg, .../* (char  *) NULL */);printf("before execl\n");if(execlp("ls","ls","-l",NULL) == -1)//不用找到命令的路径{printf("execl failed!\n");perror("why");}printf("after execl\n");return 0;
}

运行结果:
在这里插入图片描述

通过whereis指令查找ls、date(获取系统时间)系统指令的位置:

在这里插入图片描述
在这里插入图片描述

execv execvp

int main(){//int execv(const char *path, char *const argv[]);      char* argv[]={"ls",NULL,NULL};if(execv("/bin/ls",argv) == -1){printf("execl failed!\n");perror("why");}printf("after execl\n");return 0;
}
int main(){// int execvp(const char *file, char *const argv[]);char* argv[]={"ls","-l",NULL};if(execvp("ls",argv) == -1){printf("execl failed!\n");perror("why");}printf("after execl\n");return 0;
}

exec配合fork使用

应用场景:在执行A程序的过程中让子进程去执行B程序

代码B(用来修改配置文件) 编译生成程序名为change

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/wait.h>int main()
{int fd;int n_write;int n_read;pid_t pid;char *readBuf;fd=open("test15.txt",O_RDWR);int size=lseek(fd,0,SEEK_END);//计算文件的大小lseek(fd,0,SEEK_SET);//光标重新到开头readBuf=(char *)malloc(sizeof(char)*size+8 );//+8防止溢出n_read=read(fd,readBuf,size);char *p=strstr(readBuf,"length=");if(p == NULL ){printf("没有找到\n");exit(-1);}p=p+strlen("length=");//指针往后移动大哦想要改的地方*p='9';//将里面的6改称9lseek(fd,0,SEEK_SET);n_write=write(fd,readBuf,strlen(readBuf));close(fd);exit(0);
}

代码A

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/wait.h>int main()
{int fd;int n_write;int n_read;pid_t pid;int data;char *readBuf;while(1){printf("请输入一个数\n");scanf("%d",&data);if(data == 1){printf("成功进入\n");pid=fork();if(pid > 0){//父进程wait(NULL);//防止子进程变成僵尸进程}if(pid == 0){//子进程execl("./change","change",NULL,NULL);//获取系统时间也能实现}}else{printf("等待输入正确的指令\n");}}return 0;
}

初始的test15.txt
在这里插入图片描述

运行程序A后

在这里插入图片描述

system函数

本质上是对execl函数的二次封装,可查看其源码。实际上比execl更好用
和execl区别:system执行完该函数后,还会继续执行后面的函数 ,而exec则不会。

#include <stdlib.h>
int system(const char *command);
system()函数返回值如下:
成功,则返回进程的状态值
当sh不执行时,返回127//shell脚本
失败返回 -1
#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;pid_t repid;int data;while(1){printf("please input a number:");scanf("%d",&data);if(data==1){repid=fork();//创建进程if(repid > 0){wait(NULL);}if(repid == 0){//execl("./changeNumToFile.out","changeNumToFile.c","changeNum.c",NULL);system("./changeNumToFile.out");//执行完该函数后,还会执行后面的函数  exec就不会回来继续执行后面代码了,除非execl出错返回-1exit(0);}}else{printf("waiting!\n");}}return 0;

popen函数

比system在应用中的好处:可以获取运行的输出结果

#include <stdio.h>
FILE *popen(const char *command, const char *type);
int pclose(FILE *stream);
#include <stdio.h>
#include <stdlib.h>//      FILE *popen(const char *command, const char *type);
//       FILE *fopen(const char *path, const char *mode);
//      size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
int main(){//      system("ls");FILE* fp;char ret[1024];fp=popen("ls -l","r");int n_read=fread(ret,1,1024,fp);if(n_read != -1){printf("read sucess!\n");printf("return n_read=%d,read data=%s\n",nread,ret);}else{printf("no read datas\n");}return 0;
}

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

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

相关文章

简练软考知识点整理-控制成本过程

控制成本是监督项目状态&#xff0c;以更新项目成本&#xff0c;管理成本基准变更的过程。本过程的主要作用是&#xff0c;发现实际与计划的差异&#xff0c;以便采取纠正措施&#xff0c;降低风险。要更新预算&#xff0c;就需要了解截至目前的实际成本。只有经过实施整体变更…

求一列数据中的波峰_PowerQuery:横向/纵向追加数据

上一篇文章都是在原表数据基础上的分分合合&#xff0c;但做数据分析的时候还经常需要在原有数据的基础上增加一些辅助数据&#xff0c;比如加入新列、新行&#xff0c;或者从其他表中添加进来更多维度的数据&#xff0c;这些就是数据丰富的过程。01添加列Power Query中添加列有…

HTML对字体的操作详解

摘自&#xff1a;HTML对字体的所有操作详解&#xff08;经典&#xff09; 作者&#xff1a;HeroKern 发布时间&#xff1a; 2016-01-31 21:15:31 网址&#xff1a;https://blog.csdn.net/qq_21792169/article/details/50615919/?utm_termhtml%E6%A0%87%E8%AE%B0%E5%AD%97%E4%B…

数学学习笔记-三角函数

1.圆的一周的弧度数为2π&#xff0c;360角2π弧度&#xff0c;1为π/180弧度 2.如下图&#xff0c;在一个直角三角形中 角A的对边为正对着的那个边a角A的邻边为另外一条直角边b角A的斜边为斜边c其中 正弦sin(A)对边/斜边余弦cos(A)邻边/斜边正切tan(A)对边/邻边正割csc(A)1/si…

云计算呼叫中心_SaaS云呼叫中心系统只用于销售或客服?

随着时代的发展&#xff0c;公司企业不一定有规模了才需要呼叫中心系统。SaaS云部署方式呼叫中心系统的出现&#xff0c;已经能够满足所有公司的需要&#xff0c;成本低、功能完善、效率高。公司电话管理系统通常大家的理解是&#xff0c;呼叫中心只是应用于针对营销部门或是客…

mvc4 html.pager,MVC分页之MvcPager使用详解

最近刚刚接触MVC不久&#xff0c;因项目中要用到分页&#xff0c;网上找了下资料&#xff0c;最后采用了MvcPager(http://www.webdiyer.com/),支持同步和Ajax异步分页。废话不多说了直接上代码。一.MvcPager异步 ViewModel:public class Article{[Display(Name "信息编号…

原生希望原生JavaScript开篇

本篇文章个人在深圳游玩的时候突然想到的...最近就有想写几篇关于原生希望的文章&#xff0c;所以回家到之后就奋笔疾书的写出来发布了 一直对前端技巧很有兴致&#xff0c;就心生了写一个专栏的动机&#xff0c;然后就申请了原生JavaScript这个专栏&#xff0c;旨在与大家同共…

POJ 1836 Alignment

有一排人&#xff0c;身高可能不同&#xff0c;现在有一个理想状态是这排的每个人向左或向右看没有被挡住视野(当遇到等高或更高的人时会被挡住)&#xff0c;现在问最少让几人出列可以达到这个理想状态。 最少人出列&#xff0c;其实就是一个人数最多的理想状态。求一个人数最多…

ppt倒计时_年终会议做一个这样的倒计时PPT,保证惊艳全场!1分钟就能学会

倒计时动画很经常被用在一些产品的发布会或是新年晚会的现场&#xff0c;因为倒计时可以营造除以中紧张的氛围~那么我们常见的倒计时动画都是怎么做出来的呢&#xff1f;其实一点也不难&#xff0c;不需要任何专业的视频软件&#xff0c;只要用我们日常工作中最常用的PPT就能做…

修改Advance Template Jsp模板的编码格式

2019独角兽企业重金招聘Python工程师标准>>> windows ——》 preferences ——》MyEclipse——》Files and Editers——》JSP 在此widzard中修改相应的编码。 转载于:https://my.oschina.net/wangfree/blog/127313

ASP.NET Core ---日志

一、日志记录&#xff1a; 1、日志的作用&#xff1a; 程序中记录日志一般有两个目的&#xff0c;故障定位和显示程序运行状态。好的日志记录方式可以提供足够多定位问题的依据。 2、日志的等级&#xff1a; 有良好工作习惯的人&#xff0c;工作的时候会将领导交待下来的工作分…

circle函数用法 turtle_Python绘图库Turtle详细分析

关注Python学习交流学习更多Python知识Turtle库是Python语言中一个很流行的绘制图像的函数库&#xff0c;想象一个小乌龟&#xff0c;在一个横轴为x、纵轴为y的坐标系原点&#xff0c;(0,0)位置开始&#xff0c;它根据一组函数指令的控制&#xff0c;在这个平面坐标系中移动&am…

USB主机是如何检测到设备的插入的呢?

USB设备的插入检测机制 首先&#xff0c;在USB集线器的每个下游端口的D和D-上&#xff0c;分别接了一个15K欧姆的下拉电阻到地。这样&#xff0c;在集线器的端口悬空时&#xff0c;就被这两个下拉电阻拉到了低电平。而在USB设备端&#xff0c;在D或者D-上接了1.5K欧姆上拉电阻.…

单招计算机专业考多少分可以录取,单招考多少分能过?单招分数线

单独招生是高等职业院校的一种招生形式&#xff0c;和普通高考相比&#xff0c;虽然单招人数连年增加&#xff0c;但报考人数相比高考总人数还是略少的&#xff0c;竞争压力没有那么大&#xff0c;录取率相对高很多&#xff0c;基本不存在落榜&#xff0c;上大学更安全&#xf…

全国计算机等级考试题库二级C操作题100套(第97套)

更多干货推荐可以去牛客网看看&#xff0c;他们现在的IT题库内容很丰富&#xff0c;属于国内做的很好的了&#xff0c;而且是课程刷题面经求职讨论区分享&#xff0c;一站式求职学习网站&#xff0c;最最最重要的里面的资源全部免费&#xff01;&#xff01;&#xff01;点击进…

【job】2013年5-5阿里巴巴暑期实习招聘笔试题目及部分答案

网上各种标为2013年&#xff0c;实际上都是2012年或者更早的&#xff0c;下面的才是真正的2013年5月5日考试的卷子。 答题说明&#xff1a; 1.答题时间90分钟&#xff0c;请注意把握时间&#xff1b; 2.试题分为四个部分&#xff1a;单项选择题&#xff08;10题&#xff0c;20分…

Linux进程间通信(管道、消息队列、共享内存、信号、信号量)

目录Linux进程间通信概述1.管道无名管道(pipe)有名管道(fifo)2.消息队列(msg)消息队列的通信原理消息队列相关api消息队列收发数据键值生成消息队列移除3.共享内存(shm)4.信号(sig)信号概述信号编程&#xff08;入门&#xff09;信号携带消息&#xff08;高级&#xff09;5.信号…

Eclipse上GIT插件EGIT使用手册之五_查看历史记录

Team -> Show in history可以查看版本历史提交记录 可以选择对比模式

solr7.4 centos7安装

环境&#xff1a;centos7、JDK1.8、solr 自带Jetty启动 一、安装JDK1.8环境 1、下载JDK jdk-8u172-linux-x64.rpm 下载地址&#xff1a;http://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html 2、安装JDK rpm -ivh jdk-8u131-linux-x64.rpm 3、…

How to Fix an App that Crashes in Release but n...

2019独角兽企业重金招聘Python工程师标准>>> 今天在测试app时&#xff0c;发现app store上版本有奔溃现象&#xff0c;但是直接用xcode跑本地程序没问题。猜测release&debug版本造成的&#xff0c;后发现targets的 optimization level设置问题&#xff0c;将rel…