fork()
函数是一个创建新进程的系统调用,通常在 UNIX 和类 UNIX 操作系统中使用。该调用会创建当前进程的一个副本,形成两个完全相同但独立运行的进程。下面详细解释 fork()
函数的一些关键点:
-
返回值:
- 在父进程中,
fork()
返回新创建子进程的进程 ID(PID)。 - 在子进程中,
fork()
返回 0。 - 如果
fork()
调用失败,返回值小于 0,表示出错。
- 在父进程中,
-
父子进程的执行:
- 在
fork()
调用后,父子进程在调用点之后的代码部分开始独立执行。 - 父子进程执行的顺序和时间是不确定的,由操作系统的调度器决定。
- 在
-
相同的代码和数据:
- 父子进程共享代码段(text segment)和只读数据段(read-only data segment)。
- 子进程会获得父进程数据段(data segment)和堆栈(stack)的副本,但是这些是相互独立的,一个进程的修改不会影响另一个。
-
父子进程的区别:
- 最明显的区别是它们有不同的 PID。
fork()
在子进程中返回 0 是一个用于判断当前是父进程还是子进程的关键。
-
文件描述符:
- 在
fork()
之后,父子进程共享同样的打开文件描述符。如果父进程关闭一个文件描述符,而子进程仍然使用该文件描述符,会导致子进程的文件描述符关闭。
- 在
-
僵尸进程:
- 当子进程比父进程先结束时,子进程的退出状态会保持在系统中,等待父进程调用
wait()
或waitpid()
来获取。这时子进程成为僵尸进程(Zombie Process)。
- 当子进程比父进程先结束时,子进程的退出状态会保持在系统中,等待父进程调用
-
错误处理:
- 调用
fork()
后,应当检查返回值以判断调用是否成功。
- 调用
/*
#include<sys/types.h>
#include<unistd.h>
pid_t fork(void):
函数作用: 用于创建子进程
返回值:
fork会返回两次,一次在父进程,一次在子进程
在父进程中返回子进程的PID
在子进程返回0
如何区分父进程和子进程:看返回值
在父进程中返回-1,表示创建子进程失败,并设置errno
*/
#include <sys/types.h>
#include <unistd.h>
#include <stdio.h>
int main() {
// 这段代码会在系统父进程和子进程中都执行
pid_t pid = fork();
if (pid > 0) {
// Parent process
printf("I am the parent process. PID: %d, PPID: %d\n", getpid(), getppid());
} else if (pid == 0) {
// Child process
printf("I am the child process. PID: %d, PPID: %d\n", getpid(), getppid());
} else {
// Fork failed
fprintf(stderr, "Fork failed\n");
return 1;
}
// Both parent and child processes will execute the following loop
for (int i = 0; i < 3; i++) {
printf("PID: %d\n", getpid());
}
return 0;
}