过去一年一直在看操作系统原理,但是只是看,而没有实际动手去做,导致的结果是现在还不会多进程和多线程的开发,在软件开发这个领域,必须要动手去写代码的,不然永远也学不会软件开发,所以,我写下我做的点滴,也算是和大家分享交流。
# include <unistd.h> # include <sys/types.h> pid_t fork(void)
# include <unistd.h> # include <sys/types.h> # include <stdio.h> int main(void) { pid_t pid; printf("Process Creation Study \n"); pid = fork(); switch(pid) { case 0: printf("Child process is running, CurPid is %d, ParentPid is %d\n",pid,getppid()); break; case -1: printf("Process Creation faild\n"); break; default: printf("Parent process is running,ChildPid is %d,Parentpid is %d\n",pid,getppid()); break; } exit(0); }
输出结果:
fork函数非常特殊,一次调用两次返回,也就是说,它有两个返回值,即调用一次,返回两次。
返回两次的前提是进程创建成功。
一次是父进程调用fork函数的返回值,该返回值是刚刚创建的子进程的ID
一次是子进程中的fork函数的返回值,返回0
如果创建失败返回-1
特别提示:fork之后是子进程先执行还是父进程先执行是确定的,这是由内核所用的调度算法决定的。
稍微修改下,让子进程和父进程交替执行。
# include <unistd.h> # include <sys/types.h> # include <stdio.h> int main(void) { pid_t pid; char* msg; int k; printf("Process Creation Study \n"); pid = fork(); switch(pid) { case 0: msg = "Child process is running"; k = 3; break; case -1: perror("Process Creation faild\n"); break; default: msg = "Parent process is running"; k = 5; break; } while(k>0) { puts(msg); sleep(1); k--; } exit(0); }
对于输出结果:我们用图说话
通常fork失败的原因是父进程拥有的子进程的个数超过了规定的限制,此时error为EAGAIN
如果是可供内存不足也会导致进程创建失败,此时的error值为ENOMEN
总结:
1. 子进程有自己唯一的ID
2.fork的返回值不同,父进程返回的是子进程的ID,子进程返回的则为0
3.不同的父进程ID,子进程的父进程ID为创建它的父进程的ID
4.子进程共享父进程打开文件描述符,但父进程对文件描述符的改变不好影响子进程中的文件描述符。
5.子进程不继承父进程设置的文件锁
6.子进程不继承父进程设置的警告
7.子进程的未决信号集被清空
关于进程创建还有一个vfork(),我会在随后的文章里面详细介绍这个函数。