unix多进程编程之fork函数笔记
运行环境:mac+xcode
fork进程介绍
fork函数是复制本进程的一个拷贝,生成该进程的子进程。
两个进程相同的地方:
- 和该进程相关联的全部数据(包括变量,内存空间,缓冲区,文件描述符等等)
- 程序的执行上下文(execution context),两个进程会从fork后的相同位置开始继续执行
两个进程主要不同的地方:
- pid=fork();中的pid不同,父进程的pid是子进程的系统pid,子进程的pid是0。
简单示例程序
#include "stdio.h"
#include "unistd.h"
int main() {
pid_t pid;
int count = 0;
printf("before fork process\n");
pid = fork();
if (pid == 0) {
count++;
printf("child process\n");
printf("count = %d\n", count);
}
else if(pid > 0) {
//waitpid(NULL);
count++;
printf("parent process\n");
printf("count = %d\n", count);
}
else {
printf("fork failed\n");
}
return 0;
}
串口输出
“`
before fork process
parent process
count = 1
child process
count = 1
““
这里说明,虽然有“parent process”和“child process”打印,但是实际上是各个分支分别在两个不同的进程执行了一遍
在语句pid=fork()之前,只有一个进程在执行这段代码,但在这条语句之后,就变成两个进程在执行了,这两个进程的几乎完全相同,将要执行的下一条语句都是if(fpid<0)……
先看下fork的返回值:
- 在父进程中,fork返回新创建子进程的进程ID;
- 在子进程中,fork返回0;
- 如果出现错误,fork返回一个负值;
在fork函数执行完毕后,如果创建新进程成功,则出现两个进程,一个是子进程,一个是父进程。在子进程中,fork函数返回0,在父进程中,fork返回新创建子进程的进程ID。我们可以通过fork返回的值来判断当前进程是子进程还是父进程。
最终子进程和父进程执行结束
本次笔记简单记录到这里
PS
一、fork出错可能有两种原因:
- 当前的进程数已经达到了系统规定的上限,这时errno的值被设置为EAGAIN。
- 系统内存不足,这时errno的值被设置为ENOMEM。
二、不要随便fork进程,尤其是不注意区分子进程和父进程的时候,很容易出现预想不到的额外fork出新的进程
三、fork进程通常有两种用法:1.调用fork()之后立即调用exec()执行新的程序,生成一个全新的进程。2.调用fork()之后不调用exec(),仅仅是为将当前进程生成多个,以提升软件并发能力——典型的是Web Server,如Apache,nginx等。