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,一经查实,立即删除!

相关文章

亚马逊云主机AWS-EC2建站简介

其实挺多是参考互联网的&#xff0c;所以直接贴网址了。 【申请帐号】要信用卡号码的 http://bbs.admin5.com/thread-9440113-1-1.html 【地区测速】http://www.cloudping.info/ 【安装系统及SSH】要看清免费的项目 http://blog.chinaunix.net/uid-26726420-id-3196899.html 【…

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

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

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

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

二级菜单HTML原理,CSS多级菜单的实例代码讲解

这是一个相当炫的功能&#xff0c;让网页看起来像桌面程序&#xff0c;如window的开始菜单。实现原理基本和纯CSS相册差不多&#xff0c;但要注意的事项比较多&#xff0c;让我们一步步来吧。先来一个非常简单的一级菜单与悬停效果。菜单一菜单二菜单三菜单四结构很熟悉吧&…

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…

Discuz初探

根目录文件 admin.php 后台入口文件 api.php 合作API输出 archiver 论坛Archiver阅读模式&#xff08;无图版&#xff09; connect.php 云平台接口文件 cp.php 多应用服务入口文件(加载userapp.php) crossdomain.xml fa…

数学学习笔记-三角函数

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;呼叫中心只是应用于针对营销部门或是客…

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

第91套&#xff1a; 函数fun的功能是&#xff1a;计算请在程序的下划线处填入正确的内容并把下划线删除&#xff0c;使程序得出正确的结果。 注意&#xff1a;源程序存放在考生文件夹下的BLANK1.C中。 不得增行或删行&#xff0c;也不得更改程序的结构&#xff01; 给定源程序…

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

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

lnmp压力测试

最近在自己的台式机上搭了个lnmp&#xff0c;想用压力压一压&#xff0c;看看结果怎么样。由于nginx只是个静态web服务器&#xff0c;并发能力超强&#xff0c;因此lnmp的性能瓶颈在于php的处理能力&#xff0c;即php-fpm。硬件CPU AMD Phenom(tm) II X4 945内存 4G x 4硬盘 西…

1 uC/OS工程目录

第一个项目&#xff0c;从整体上认识 uC/OS转载于:https://www.cnblogs.com/chdfelix/p/9581271.html

html中免费的四级联动,利用JS实现省市区街道四级联动插件

特效描述&#xff1a;利用JS实现 省市区街道 四级联动插件。利用JS实现省市区街道四级联动插件代码结构1. 引入CSS2. 引入JS3. HTML代码所在地区所在地区请选择请选择请选择请选择$(function(){// 地址选择器遮罩层打开与关闭$("#areaLabel").click(function(e){$(&q…

python递归面试题_汉诺塔问题其实很简单 Python 递归经典面试题

话不多说&#xff0c;上代码1 def hanoi_move(n, source, dest, intermediate):2 if n > 1: # 递归出口&#xff0c;只剩一个盘子3 hanoi_move(n-1, source, intermediate, dest)4 print("Move %s -> %s" % (source, dest))5 hanoi_move(n-1, intermediate, de…

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

第92套&#xff1a; 函数fun的功能是&#xff1a;计算的前n项。若x2.5&#xff0c;函数值为&#xff1a;12.182340。 请在程序的下划线处填入正确的内容并把下划线删除&#xff0c;使程序得出正确的结果。 注意&#xff1a;源程序存放在考生文件夹下的BLANK1.C中。 不得增行或…

原生希望原生JavaScript开篇

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

POJ 1836 Alignment

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

2021年广西高考成绩查询方法,2021年广西高考成绩查询网站查分网址:https://www.gxeea.cn/...

【摘要】高考结束后大家最为关心的问题就是在哪里查分&#xff0c;如何查分&#xff0c;高考频道特别整理2021年广西高考成绩查询查分网址&#xff0c;成绩公布时考生可直接点击网址进行查分&#xff0c;预祝大家都能顺利的考上理想的大学&#xff01;由于&#xff0c;各省级招…

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

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

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

第93套&#xff1a; 给定程序中已建立一个带有头结点的单向链表,在main函数中将多次调用fun 函数,每调用一次fun函数&#xff0c;输出链表尾部结点中的数据&#xff0c;并释放该结点&#xff0c;使链表 缩短。 请在程序的下划线处填入正确的内容并把下划线删除&#xff0c;使程…