目录
- 写在前面
- 关于fork()的返回值,参照man 2 fork
- 案例,更好的运用fork(),这里有两种经典用法,这里给出第一种,第二种和exec函数族一起写。详见:
https://blog.csdn.net/weixin_39956356/article/details/86624472
(一)写在前面
(1):操作系统是并发的,多任务,在一个时间片,多个进程轮流使用cpu,要打破单片机只能执行一个程序!要有并发的思想,不然你感觉代码很诡异。
(2):这次分享的是–fork的运用,自己不深入剖析fork,对于你初次接触,能运用好就行,如果你要深究里面是怎么运行的,有些不合适,庞杂Linux是要慢慢积累,有个过程的。
(二)fork()的返回值
只有三句话:有必要给出英语的原文
成功:
(1) the PID of the child process is returned in the parent. 子进程pid返回给父进程
(2) and 0 is returned in the child. 而0返回给子进程
失败:
(3) On failure, -1 is returned in the parent.失败,-1返回给父进程。内存不够或者id 号用尽,不过这种情况几乎很少发生。
(4)其实,网上有大佬把fork理解成链表,父进程指向子进程,而子进程是0表示结束,这样就不会忘记父子进程的返回值了
(三)案例,务必理解
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
/****************************************************************************************
** 创建子进程create a child process
** fork():
** pid_t fork(void);
** RETURN VALUE
** the PID of the child process is returned in the parent
and 0 is returned in the child.
On failure, -1 is returned in the parent
即成功情况下:子进程pid返回给父进程,而0返回给子进程
即失败情况下:-1返回给父进程。内存不够或者id 号用尽,不过这种情况几乎很少发生。
** 注意:虽然else中两个i名字一样,但是并不是同一个;两个的地址并不一样,这是因为调用fork后父子进程几乎有一样的资源
****************************************************************************************/
int main()
{
pid_t pid;
int i = 100;
pid = fork();
//创建fork出错
if(pid == -1)
{
printf("Creat fork error!!!\n");
exit(1);
}
//父进程
else if(pid)
{
i++; //两个i地址并不相同,你可以把它想成C语言的作用域(帮助理解,实则不是这样的)
printf("The father i = %d.\n",i);
printf("The father return %d.\n",pid); //子进程号返回给父进程
printf("The father pid is %d.\n",getpid()); //父进程的pid
printf("The father ppid is %d.\n",getppid()); //父进程的父进程pid
while(1); //这里用while(1);因为父、子进程是并发了,父进程退出了就看不到打印信息了
}
//子进程
else
{
i++; //两个i地址并不相同,你可以把它想成C语言的作用域(帮助理解,实则不是这样的)
printf("\nThe child i = %d.\n",i);
printf("The child return %d.\n",pid); //0返回给子进程
printf("The child pid is %d.\n",getpid()); //子进程的pid
printf("The child ppid is %d.\n",getppid());//子进程的父进程pid
while(1); //这里用while(1);因为父、子进程是并发了,子进程退出了就看不到打印信息了
}
return 0;
}
输出:
分析:
(四)重点
- 调用fork函数,会创建一个新的进程,它几乎和父进程有差不多的资源–上述两个i是不一样的,这些是由操作系统完成的
- 调用fork函数,父、子进程并发运行,若果不采取相应措施,得不到想要的结果
- 子进程的pid 和父进程不一样,子进程的ppid 会设置为父进程的pid
- 子进程中的资源统计信息会清零
- 所有文件锁也不会被子进程继承
- 挂起的信号会被清除,也不会被继承
- 用法二的框架: