先统一把需要用到的头文件贴出来
#include<stdlib.h>
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/wait.h>
#include<signal.h>
不考虑创建进程失败的情况,因此fork()函数返回值只有0或1两种。返回值为0时是子进程,返回值为1时是父进程
1.首先是一个最简单的调用fork的函数:fork0()
void fork0()
{
if(fork() == 0)
printf("Hello from child\n");
else
printf("Hello from parent\n");
}
在这个函数中,调用了一次fork(),因此共有两个进程。
进程图如下:
两次分别输出各自的语句信息,我运行的情况是parent总比child先输出,如果想先输出child,可以使用sleep()让父进程休眠一会。
输出结果:
Hello from parent
Hello from child
2.计算变量值:fork1()
void fork1()
{
int x = 1;
pid_t pid = fork();
if(pid==0)
printf("Child has x = %d\n", ++x);
else
printf("Parent has x = %d\n", --x);
printf("Bye from process %d with x = %d\n", getpid(), x);
}
进程图如下:
x初始值为1,子进程执行++x,x的值变为2;父进程执行- -x,x的值变为0。两个进程独立进行,相互不受影响。
输出结果:
Parent has x = 0
Bye from process 2183 with x = 0
Child has x = 2
Bye from process 2184 with x = 2
3.调用两次fork():fork2()
void fork2()
{
printf("L0\n");
fork();
printf("L1\n");
fork();
printf("Bye\n");
}
进程图如下:
调用第一个fork()时,共有2个进程,打印输出L1。此时的父进程和子进程都各自调用fork(),因此共有4个进程,打印输出Bye。注意在输出L1之前,不可能输出Bye,但之后的输出顺序可能是混乱的。
输出结果:
L0
L1
Bye
Bye
L1
Bye
Bye
4.使用了两个if语句,在if语句中调用fork()函数:fork3()
void fork3()
{
printf("L0\n");
if(fork()!= 0){
printf("L1\n");
if(fork()!= 0){
printf("L2\n");
}
}
printf("Bye\n");
}
进程图如下:
由于是在if语句中调用fork()函数,每次每次进入if语句都会执行一次,因此共执行两次fork()。两次调用的意思都是只有是父进程时才输出L1或L2,因此L1只输出1次,L2只输出2次。调用两次fork()后共有4个进程,每次都输出Bye。
输出结果:
L0
L1
L2
Bye
Bye
Bye
5.父进程一直在循环,无法在子进程结束后将其回收:fork4():
void fork4()
{
if(fork() == 0){ /* Child */
printf("Terminating Child, PID = %d\n", getpid());
exit(0);
}
else{
printf("Running Parent, PID = %d\n", getpid());
while(1) /* Infinite loop */
;
}
}
进程图如下:
子进程正常运行结束后等待父进程将它回收,但父进程一直在做while循环没有结束,所以在父进程和子进程都打印输出后程序不会结束运行,会一直等待,只能用Ctrl+z强制结束进程。
输出结果:
用ps参数来查看进程状态时,会发现强制结束后的进程并没有没回收:
6.fork5()
#define N 5
void fork5()
{
pid_t pid[N];
int i, child_status;
for (i = 0; i < N; i++)
if((pid[i] = fork()) == 0){
exit(100+i); /* Child */
}
for(i = 0; i < N; i++) { /* Parent */
pid_t wpid = wait(&child_status);
if (WIFEXITED(child_status))
printf("Child %d terminated with exit status %d\n",wpid, WEXITSTATUS(child_status));
else
printf("Child %d terminate abnormally\n", wpid);
}
}
进程图如下:
如果子进程是正常结束,则输出进程号及返回数,否则提示异常。fork()创建子进程后,子进程结束顺序不定,会无规则输出子进程结束语句。
输出结果:
目前只选择了6个有关fork()函数用法的代码文件,以后学得更熟练了应该还会继续更新。