- fork创建一个子进程的一般目的:
(1)一个父进程希望复制自己,使父、子进程同时执行不同的代码段。这在网络服务进程中是常见的——父进程等待委托者的服务请求。当这种请求到达时,父进程调用 fork,使子进程处理此请求。父进程则继续等待下一个服务请求。
(2) 一个进程要执行一个不同的程序。这对shell是常见的情况。在这种情况下,子进程在从fork返回后立即调用exec(我们将在8 . 9节说明exec)。
一、创建进程的实际应用场景:
code:demo8.c
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
pid_t pid;
pid_t pid2;
int data;
while(1){
printf("Please input a data:\n");
scanf("%d",&data);
if(data == 1){
pid=fork();
pid2=getpid();
if(pid > 0){
printf("pid:%d\n",getpid());
}
else if(pid == 0){
while(1){
printf("connect succces,pid:%d\n",pid2);
sleep(2);
}
}
}
else{
printf("No thing,wait connect...\n");
}
}
return 0;
}
运行结果:(每输入1,创建一个子进程)
CLC@Embed_Learn:~/LinuxLearn/PID$ ./a.out
Please input a data:
1
pid:43789
Please input a data:
connect succces,pid:43790
connect succces,pid:43790
connect succces,pid:43790
connect succces,pid:43790
connect succces,pid:43790
connect succces,pid:43790
connect succces,pid:43790
^C
CLC@Embed_Learn:~/LinuxLearn/PID$ vi demo8.c
CLC@Embed_Learn:~/LinuxLearn/PID$ vi demo8.c
CLC@Embed_Learn:~/LinuxLearn/PID$ ./a.out
Please input a data:
2
No thing,wite connect…
Please input a data:
1
pid:43924
Please input a data:
connect succces,pid:43925
connect succces,pid:43925
1
pid:43924
Please input a data:
connect succces,pid:43927
connect succces,pid:43925
1connect succces,pid:43927
connect succces,pid:43925
connect succces,pid:43927
1
No thing,wite connect…
Please input a data:
connect succces,pid:43925
connect succces,pid:43927
1
pid:43924
Please input a data:
connect succces,pid:43928
connect succces,pid:43925
connect succces,pid:43927
connect succces,pid:43928
connect succces,pid:43925
1connect succces,pid:43927
pid:43924
Please input a data:
connect succces,pid:43930
connect succces,pid:43928
connect succces,pid:43925
connect succces,pid:43927
connect succces,pid:43930
connect succces,pid:43928
^C
fork的总结
由fork创建的新进程被称为子进程(child process)。该函数被调用一次,但返回两次。两次返回的区别是子进程的返回值是 0,和父进程的返回值则是新子进程的进程 ID。将子进程ID返回给父进程的理由是:因为一个进程的子进程可以多于一个,所以没有一个函数使一个进程可以获得其所有子进程的进程 ID。fork使子进程得到返回值 0的理由是:一个进程只会有一个父进程,所以子进程总是可以调用getppid以获得其父进程的进程ID (进程ID 0总是由交换进程使用,所以一个子进程的进程I D不可能为0 )。
- 子进程和父进程继续执行f o r k之后的指令。子进程是父进程的复制品。例如,子进程获得父进程数据空间、堆和栈的复制品。注意,这是子进程所拥有的拷贝。父、子进程并不共享这些存储空间部分。如果正文段是只读的,则父、子进程共享正文段 (见7 . 6节)。
- 现在很多的实现并不做一个父进程数据段和堆的完全拷贝,因为在 fork之后经常跟随着exec。作为替代,使用了在写时复制( Copy-On-Write, COW)的技术。这些区域由父、子进程共享,而且内核将它们的存取许可权改变为只读的。如果有进程试图修改这些区域,则内核为有关部分,典型的是虚存系统中的“页”,做一个拷贝。Bach〔1986〕的9.2节和Leffler等〔1989〕的5.7节对这种特征做了更详细的说明。
关于fork详细看《UINX环境高级编程》