预备知识
首先你要清楚,进程管理为内核的功用之一。
接着我们遵循教科书中的定义:进程是程序执行时的一个实例。
再进一步解释,程序(program)为存储在硬盘上的二进制文件,而将其装载进内存之后即为进程(process),同时还将获得一个 PID(process ID)用于唯一标识该进程。
进程被产生,他们有或多或少的生命,他们也可以继续产生一个或多个子进程,但最终都要死亡。与人类不同的是,进程没有性别,每个进程都只有一个父进程。
创建新进程 fork()
fork() 通过复制调用进程的方式创建一个新的进程,新进程被称为子进程,调用进程则被称为父进程。
函数原型如下:
pid_t fork(void);
创建一个进程时,它几乎与父进程相同。就像 fork() 帮助手册上所写,子进程就是对父进程的精确复制。
The child process is an exact duplicate of the parent process
也就是说,子进程将复制父进程所拥有的资源,即拷贝父进程的整个地址空间。之后这两个进程将执行相同的程序文本,但彼此将拥有独立的栈、堆及数据。
// 代码示例
int main(int argc, char *argv[])
{
switch ( fork() ) {
case -1: // Handle error
perror("fork error");
exit(1);
case 0: // Perform actions specific to child
printf("This is child process, PID is %d\n", getpid());
printf("My PPID is %d\n", getppid());
break;
default: // Perform actions specific to parent
printf("This is parent process, PID is %d\n", getpid());
}
printf("This will be shown in both parent and child! %d\n\n", getpid());
return 0;
}
当系统调用 fork() 成功创建子进程后,将在父进程中返回子进程的 PID,并在子进程中返回 0。
因此我们才需要通过 fork() 的返回值来区分父、子进程,但 fork() 之后你不应对父、子进程的执行顺序做任何假设,这时你就需要后文的 wait() 函数。
此处需要理解的关键是,完成一次 fork() 调用后,