时间:2021-05-19
本文研究的主要是Linux进程函数fork(),vfork(),execX()的相关内容,具体介绍如下。
fork函数:创建一个新进程
1、fork()成功后,将为子进程申请PCB和用户内存空间。一个实例:
#include <unistd.h> //fork fuction#include <fcntl.h> //file operator#include <sys/types.h>#include <stdio.h> #include <stdlib.h> //exit fuction#include <string.h>int main() { pid_t pid; int i=1; int status; char *ch1="hello",*ch2="world",*ch3="IN"; int fd; if ((fd=open("fork.txt",O_RDWR|O_CREAT,0644))==-1) { perror("not open"); exit(EXIT_FAILURE); } if (write(fd,ch1,strlen(ch1))==-1) { //write in fork.txt perror("not write"); exit(EXIT_FAILURE); } if ((pid=fork())==-1) { perror("fork error"); exit(EXIT_FAILURE); } else if(pid==0) { //son process int i=2; //change i printf("child:i=%d\n",i); if (write(fd,ch2,strlen(ch2))==-1) perror("child write"); return 0; } else { sleep(1); printf("parent:i=%d\n",i); if (write(fd,ch3,strlen(ch3))==-1) perror("child write"); wait(&status); return 0; }}运行:
[root@localhost linux]# gcc -o fork fork.c [root@localhost linux]# ./fork child:i=2 parent:i=1可以看到在子进程中改变了i的值,然而父进程i仍为1,所以说子进程和父进程有自己的用户空间。而打开所创建的fork.txt可以得到hellowordIN,父子进程共同对一个文件操作写入的数据是不交叉覆盖的,说明父子进程共享文件偏移,一次共享文件表项。
与fork()函数不同,vfork()函数在创建进程是并不复制父进程的地址空间,而是在必要的时候才申请新的存储空间,因此使得vfork()更有效率。
特别注意的是vfork()是共享父进程的代码以数据段。
一个例子:
#include <unistd.h> //fork fuction#include <fcntl.h> //file operator#include <sys/types.h>#include <stdio.h> #include <stdlib.h> //exit fuction#include <string.h>int i=10;int main() { pid_t pid; if ((pid=fork())==-1) { perror("fork error"); exit(EXIT_FAILURE); } else if(pid==0) { //son process i++; printf("child:i=%d\n",i); _exit(0); } else { sleep(1); printf("parent:i=%d\n",i); return 0; }}注意:上面的代码中回收子进程用的是_exit(0),如果用return 0;的话它会回收用户空间,因此在父进程调用的时候会出现段错误。
下面是调用输出结果:
如果以fork()创建则会输出: [root@localhost linux]# ./fork child:i=11 parent:i=10 如果改为vfork(),则: child:i=11 parent:i=11用fork()函数创建紫禁城后,如果希望在当前子进程中运行新的程序,则可以调用execX系列函数。
注意:当进程调用exec函数后,该进程的用户空间资源完全有新程序代替。
这些函数的区别在于:
1、指示新程序的位置是路径还是文件名
2、在使用参数时是使用参数列表哈市使用argv[]数组
3、后缀有l(list)表示使用参数列表,v表示使用argv[]数组
具体如下所示:
#include<unistd.h>int execl(const char *pathname,const char *arg0,.../*(char *) 0 */);int execv(const char *pathname,char *const argv[]);int execle(const char *pathname,const char *arg0,.../*(char *) 0 ,char *const envp[] */);int execve(const char *pathname,char *const argv[],char *const envp[]);int execlp(const char *filename,const char*arg0,.../*(char *) 0*/);int execvp(const char *filename, char *const argv[]);int fexecve(int fd,char *const argv[],char *const evnp[]);一个实例:
#include <unistd.h>#include <stdio.h>#include <sys/types.h>int main(int argc ,char* argv[]) { pid_t pid; if ((pid=fork())==-1) printf("error"); else if (pid==0) execl("/bin/ls","ls","-l",argv[1],(char *)0); else printf("father ok\n");}运行可以看到在子进程中执行了ls命令。
[yqtao@localhost linux]$ gcc -o exec execX.c[yqtao@localhost linux]$ ./exec /home father ok//execlp()函数使用
#include <unistd.h>#include <stdio.h>#include <sys/types.h>int main(int argc ,char* argv[]) { execlp("ls","ls","-l","/home",(char*)0);}//execv()函数的使用
#include <unistd.h>#include <stdio.h>#include <sys/types.h>int main(int argc ,char* argv[]) { char* argv1[]={"ls","-l","/home",0}; execv("/bin/ls",argv1);}ecvp()会从环境变量PATH所指定的目录中查找文件名作为第一个参数,第二个及以后的参数由参数列表,注意最后一个成员必须为NULL
#include <unistd.h>#include <stdio.h>#include <sys/types.h>int main(int argc ,char* argv[]) { char* argv1[]={"ls","-l","/home",0}; execvp("ls",argv1);}以上就是本文关于深入解读Linux进程函数fork(),vfork(),execX()的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站其他相关专题,如有不足之处,欢迎留言指出。感谢朋友们对本站的支持!
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
一、php中pcntl_fork函数概述pcntl_fork()函数是php中用于创建子进程的一个函数,返回创建的子进程的pid。该函数创建子进程具体fork的
在上一篇文章中,我们讲了如何在linux上用python写一个守护进程。主要原理是利用linux的fork函数来创建一个进程,然后退出父进程运行,生成的子进程就
linux下我们可以调用fork函数创建子进程,创建的子进程将会得到父进程的数据空间、堆、栈......副本(采用写时复制机制),子进程将会继承父进程的信号掩码
Linux进程控制详解及实例常用函数:fork()通过复制调用进程来建立新的进程,是最基本的进程建立操作。exec包括一系列的系统调用,其中每个系统调用都完成相
fork()函数用于从已存在的进程中创建一个新进程。新进程称为子进程,而园进程称为父进程。使用fork()函数得到的子进程是父进程的一个复制品,它从父进程处继承