Fork同时建立多个子进程方法html
第一种方法:验证经过 linux
特色:同时建立多个子进程,每一个子进程能够执行不一样的任务,程序 可读性较好,便于分析,易扩展为多个子进程 函数
#include <stdio.h> #include <pthread.h> int main(void) { printf("before fork(), pid = %d\n", getpid()); pid_t p1 = fork(); if ( p1 == 0 ) { printf("in child 1, pid = %d\n", getpid()); return 0; //若此处没有return 0 p1 进程也会执行 pid_t p2=fork()语句 } pid_t p2 = fork(); if ( p2 == 0 ) { printf("in child 2, pid = %d\n", getpid()); return 0; //子进程结束,跳回父进程 Printf("hello world\n"); //没有打印 } int st1, st2; waitpid( p1, &st1, 0); waitpid( p2, &st2, 0); printf("in parent, child 1 pid = %d\n", p1); printf("in parent, child 2 pid = %d\n", p2); printf("in parent, pid = %d\n", getpid()); printf("in parent, child 1 exited with %d\n", st1); printf("in parent, child 2 exited with %d\n", st2); return 0; }
[root@localhost ~]# gcc -o thread -pthread t.c
[root@localhost ~]# ./thread
before fork(), pid = 2629
in child 1, pid = 2630
in child 2, pid = 2631
in parent, child 1 pid = 2630
in parent, child 2 pid = 2631
in parent, pid = 2629
in parent, child 1 exited with 0
in parent, child 2 exited with 0url
第二种方法: 验证经过 .net
特色:同时建立两个子进程,结构比较繁琐,程序可读性很差,不易扩展 线程
#include<stdio.h> #include<unistd.h> #include<sys/types.h> //这个头文件不能少,不然pid_t没有定义 main() { printf("This is parent process%d\n",getpid()); pid_t p1,p2; if((p1=fork())==0) { printf("This is child_1 process%d\n",getpid()); }else { if((p2=fork())==0) { printf("This is child_2 process%d\n",getpid()); }else { wait(p1,NULL,0); wait(p2,NULL,0); printf("This is parent process%d\n",getpid()); } } }
第三种方法:for 循环方法 code
特色:其实每次循环只是建立了单个进程,并无同时建立多个进程 htm
#include<stdio.h> #include<unistd.h> #include<sys/types.h> main() { printf("This is parent process%d\n",getpid()); pid_t p1,p2; int i; for(i=0;i<=2;i++) { if((p1=fork())==0) { printf("This is child_1 process%d\n",getpid()); return 0;//这个地方很是关键 } wait(p1,NULL,0); //父进程等待p1子进程执行后才能继续fork其余子进程 printf("This is parent process%d\n",getpid()); } }
注意:标注的 return 0 对程序结果影响很大 blog
无 return 0 状况 递归
#include<stdio.h> #include<unistd.h> #include<sys/types.h> main() { printf("This is parent process%d\n",getpid()); pid_t p1,p2; int i; for(i=0;i<=2;i++) { if((p1=fork())==0) { printf("This is child_1 process%d\n",getpid()); //return 0;//这个地方很是关键 } wait(p1,NULL,0); printf("This is parent process%d\n",getpid()); } }
结论:父进程会生成 n(n+1)/2+1个子进程,N 为循环次数,本例中共有 7 个子进程, 但实际上只有 3 个是父进程产生的,其他都为子进程 fork()出来的。父进程fork了3个进程,第一个子进程执行完以后又fork了2个进程,第2个子进程fork了1个进程。
正确的使用Linux中的用fork()由一个父进程建立同时多个子进程 的格式以下:
int status,i; for (i = 0; i < 10; i++) { status = fork(); if (status == 0 || status == -1) break;//每次循环时,若是发现是子进程就直接从建立子进程的循环中跳出来,不让你进入循环,这样就保证了每次只有父进程来作循环建立子进程的工做 } if (status == -1) { //error } else if (status == 0) //每一个子进程都会执行的代码 { //sub process } else { //parent process }
原文地址:http://www.cnblogs.com/hanyan225/archive/2011/07/22/2113606.html
怎么建立多个进程呢?我说那还不容易,看下边代码:
//省略必要头文件 int main() { pid_t pid[2]; int i; printf("This is %d\n",getpid()); for(i = 0;i < 2;i++ ){ if((pid[0] = fork()) < 0){ printf("Fork() Error!"); exit(-1); } if(pid[0] == 0) printf("This is parent %d,child is %d\n",getppid(),getpid()); else wait(5); } return 0; }
好,这段代码仍是挺简单的,咱们的意思是:主线程经过循环建立2个子进程,这时系统中的总进程数应该是3,看看输出结果吧:
这下你明白了吧,问题没有想象中的那样简单,父进程如今标号为1的循环中创了一个子进程,而后第二次循环,前边的第一个子线程又建立一个子进程,这时明显系统中有四个进程,仍是不懂?在下边的时序图吧:
这下你应该明白了吧,好了问题知道了,怎么解决,方法有二;
方法一:直接看代码 for循环
void createsubprocess(int num) { pid_t pid; int i; for(i=0;i<num;i++) { pid=fork(); if(pid==0||pid==-1) //子进程或建立进程失败均退出,这里是关键所在 { break; } } if(pid==-1) { perror("fail to fork!\n"); exit(1); } else if(pid==0) { printf("子进程id=%d,其对应的父进程id=%d\n",getpid(),getppid()); exit(0); } else { printf("父进程id=%d\n",getpid()); exit(0); } }
这种方法的关键就在于每次循环时,若是发现是子进程就直接从建立子进程的循环中跳出来,不让你进入循环,这样就保证了每次只有父进程来作循环建立子进程的工做。
方法二:直接看代码 递归函数
void createsubprocess(int num,int max) { if(num>=max)return; pid=fork(); if(pid<0) { perror("fork error!\n"); exit(1); } //子进程 else if(pid==0) { sleep(3); printf("子进程id=%d,父进程id=%d\n",getpid(),getppid()); } //父进程 else { num++; if(num==1)printf("父进程id=%d\n",getpid()); if(num<max)createsubprocess(num,max); //此处加sleep是为了防止父进程先退出,从而产生异常 sleep(5); } }
这里的关键在于递归操做,只有父进程才进入递归建立子进程,子进程不进行这样的操做。