fork函数的功能
fork函数的作用是创建一个与父进程几乎相同的进程,它们可以同时做相同的事情,当然也可以通过初始参数的不同来做不同的事情。
当进程调用fork函数时,内核会复制所有的内部数据结构,复制进程的页表项,然后把父进程的地址空间按页复制到子进程的地址空间中,不过现在的Linux已经不会采用这种耗时的方式了,而是使用“写时复制”的方法,来避免不必要的复制。
函数原型如下:
#include <sys/types.h>
#include <unistd.h>
/*
* pid_t所对应的具体的数据类型与机器的体系结构有关,并且在任何C语言标准中没有定义它。但是
* 在Linux中,pid_t通常定义为C语言的int类型。
*/
pid_t fork (void);
当fork调用成功时,会创建一个新的进程,它几乎与调用fork()的进程完全相同。这两个进程都会继续进行。在子进程中,成功的fork()调用会返回0;在父进程中,fork()会返回子进程的pid。父进程和子进程有下面几点不同,其它方面都完全相同:
- 子进程的pid是新分配的,与父进程不同。
- 子进程的ppid会设置成父进程的pid。
- 子进程中的资源统计信息(Resource statistics)会清零。
- 所有挂起的信号都会清除,也不会被子进程继承。
- 所有文件锁都不会被子进程继承。
1出错时,不会创建子进程,fork()返回-1,并设置相应的errno值。
errno | value |
---|---|
EAGAIN | 内核申请资源失败,例如达到了进程数上限 |
ENOMEM | 内核内存不足 |
示例:
#include <sys/types.h>
#include <cstdio>
#include <cstdlib>
#include <unistd.h>
int main (int argc, char *argv[])
{
pid_t pid;
pid = fork ();
if (pid == -1)
perror ("fork");
if (!pid) {
char *args[2] = { "qtcreator", NULL };
int ret;
ret = execv ("/opt/Qt5.7.1/Tools/QtCreator/bin/qtcreator", args);
if (ret == -1) {
perror ("execv");
exit (EXIT_FAILURE);
}
}
putchar('a');
return 0;
}
程序的子进程会运行指定的应用,父进程会输出字符’a’。