大家知道Linux中创建子进程的一个很好的方法是函数调用fork,但是很多初学者对fork的理解上可能有点困难。我们举个例子来看看fork的用法吧。
大家用fork的时候记住fork是“
分叉”的意思就很好理解了。
记得初学fork() 函数及遇到这个函数时,总是不能理解fork为什么会这样写,就会分成父子两个进程。
先看一下fork() 的经典模式
先看一下fork() 的经典模式
//--------------------------------------------------------begin
int pid=fork();
if(pid < 0){
//失败,一般是该用户的进程数达到限制或者内存被用光了
........
}
else if(pid == 0){
//子进程执行的代码
......
}
else{
//父进程执行的代码
.........
}
.........
//-------------------------------------------------------end
(这段代码可能是有点问题的,稍后解释),首先来看fork 的返回值,有三种情况 -1,0,>0
-1: 当然是失败了,也不会分裂成两个进程. 返回0是子进程,返回〉0 是父进程,此时返回的值当然是资进程的pid了。 这里是比较难理解的一个地方,他的可以这样简单的解释: 当进程遇到fork调用时,将此进程整个拷贝一份,即子进程。此时,该进程的返回值被设置为〉0,即刚才 拷贝生成的子进程pid,而在拷贝生成的进程中,将返回值设置为0 。也就是此时已经有两个进程,只有pid的值不同(忽略一起其他的设置)。
int pid=fork();
if(pid < 0){
//失败,一般是该用户的进程数达到限制或者内存被用光了
........
}
else if(pid == 0){
//子进程执行的代码
......
}
else{
//父进程执行的代码
.........
}
.........
//-------------------------------------------------------end
(这段代码可能是有点问题的,稍后解释),首先来看fork 的返回值,有三种情况 -1,0,>0
-1: 当然是失败了,也不会分裂成两个进程. 返回0是子进程,返回〉0 是父进程,此时返回的值当然是资进程的pid了。 这里是比较难理解的一个地方,他的可以这样简单的解释: 当进程遇到fork调用时,将此进程整个拷贝一份,即子进程。此时,该进程的返回值被设置为〉0,即刚才 拷贝生成的子进程pid,而在拷贝生成的进程中,将返回值设置为0 。也就是此时已经有两个进程,只有pid的值不同(忽略一起其他的设置)。
此时,两个进程都从fork开始往下执行,只是pid不同,所以 if ..... else if .... esle 会根据pid不同来执行相应的代码,并不是说某一部分是父进程的代码,某一部分是自进程的代码. 只是一些条件判断而已.所以当fork后,真个代码都会被两个进程执行,只是(fork成功时)子进程中的pid 为零,所以 else if条件成立,其它两个不成立,所以执行else if中的代码.父进程中pid >0的,所以else成立,执行其中的代码. 如果for()k失败,当然是返回-1,此时是没有子进程的.
现在来看一下,fork返回值,失败,返回-1,linux下系统调用的一半惯例,错误码在errno中. 0 子进程,因为可以通过getpid() 和getppid()获取自己的进程和父进程的pid; >0 ,父进程中,此为fork()的子进程pid,因为进程中没有可以获取自己子进程pid的系统调用。此时就要把返回的pid保存起来,以后用来控制子进程。例如程序退出时调用kill(pid,9)杀死自己的子进程。
最后我想跟大家说的是,fork之后是父进程先执行还是子进程先执行,这是取决于cpu调用算法的,就是说他们谁先执行都有可能。