fork()函数用法理解

转载 2007年09月13日 12:00:00
 

#include <sys/types.h>
#include <unistd.h>
#include <stdio.h>
int main()
{
pid_t pid;
static int n = 0;
printf(”fork!/n”); /*printf(”fork!”)*/
switch (pid = fork())
{
case -1:
{
/* 这里pid为-1,fork函数失败 */
/* 一些可能的原因是 */
/* 进程数或虚拟内存用尽 */
perror(”The fork failed!”);
break;
}
case 0:
{
/* pid为0,子进程 */
printf(”[child]i am child!/n”);
printf(”[child]getpid=[%d]/n”, getpid() );
printf(”[child]pid=[%d]/n”, pid );
break;
}
default:
{
/* pid大于0,父进程 */
printf(”[parent]i am parent!/n” );
printf(”[parent]getpid=[%d]/n”,getpid() );
printf(”[parent]pid=[%d]/n”,pid );
break;
}
}
printf(”n=[%d]/n”, n++);

return 0;
}

输出结果1
fork!
[child]i am child!
[child]getpid=[4807]
[child]pid=[0]
n=[0]
[parent]i am parent!
[parent]getpid=[4806]
[parent]pid=[4807]
n=[0]
输出结果2
fork![child]i am child!
[child]getpid=[6163]
[child]pid=[0]
n=[0]
fork![parent]i am parent!
[parent]getpid=[6162]
[parent]pid=[6163]
n=[0]

如果fork成功,子进程中fork的返回值是0,父进程中fork的返回值是子进程的进程号,getpid()返回的才是各自真实的进程号。

printf(”fork!”);//print 一次; 这里会print 2次
如果将 printf(”fork!”) 换成 printf(”fork!/n”) 那么就是只打印一次了.
主要的区别是因为有了一个 /n 回车符号
这就跟Printf的缓冲机制有关了,printf某些内容时,操作系统仅仅是把该内容放到了stdout的缓冲队列里了,并没有实际的写到屏幕上
但是,只要看到有 /n 则会立即刷新stdout,因此就马上能够打印了.
运行了printf(”fork!”) 后,fork!仅仅被放到了缓冲里,再运行到fork时,缓冲里面的fork!被子进程继承了
因此在子进程度stdout缓冲里面就也有了fork!.
所以,你最终看到的会是fork!被printf了2次!!!!
而运行 printf(”fork!/n”)后,fork!被立即打印到了屏幕上,之后fork到的子进程里的stdout缓冲里不会有 AAAAAA 内容
因此你看到的结果会是 AAAAAA 被printf了1次!!!!

一段关于fork的小程序的启示


前几天,论坛上有人问了这样一个问题:
#include <sys/types.h>
#include <unistd.h>

int
main()
{

for
(int i= 0; i<3; i++)
{

int
pid= fork();
if
(pid== 0)
{

printf(“child/n”);
}

else

{

printf(“father/n”);
}
}

return
0;
}


请问输出结果是什么?

初看,想当然认为结果是3对child-father,只是顺序不确定,而且按照Unix环境高级编程中的说法,极端的情况下可能还会出现两个输出的内容相互夹杂的情况。

但是,在Unix测试了一下发现输出竟然有7对child-father。为什么会这样呢?看了半天程序终于明白了这个简单的问题。其实,这个问题在写/懂汇编的人看来是再清楚不过了,问题就出在这个for循环。
1.
i=0时,父进程进入for循环,此时由于fork的作用,产生父子两个进程(分别记为F0/S0),分别输出father和child,然后,二者分别执行后续的代码,那后续的代码是什么呢?return 0?当然不是,由于for循环的存在,后续的代码是add指令和一条jump指令,因此,父子进程都将进入i=1的情况;
2.
i=1时,父进程继续分成父子两个进程(分别记为F1/S1),而i=0时fork出的子进程也将分成两个进程(分别记为FS01/SS01),然后所有这些进程进入i=2
3.
….过程于上面类似,已经不用多说了,相信一切都已经明了了,依照上面的标记方法,i=2时将产生F2/S2,FS12/SS12,FFS012/SFS012,FSS012/SSS012.
因此,最终的结果是输出7对child/father。其对应的数学公式为:
1
+ 2+ 4+ … + 2^(n- 1) = 2^n- 1

不过话说回来,这种在for循环中使用fork的作法实在不值得推荐,研究研究尚可,实际应用恐怕会引来很多麻烦,需小心谨慎才是。

对fork函数理解

头文件: #include  pid_t fork (void); 1. 创建一个子进程,失败返回-1。 2. 调用一次,返回两次。分别在父子进程中返回子进程的PID和0。利用返回值的不同,可以分别为...
  • meetings
  • meetings
  • 2015年07月29日 11:05
  • 826

【Linux】关于理解fork()函数的简单例子

1.fork()函数fork()函数通过系统调用创建一个与原来进程几乎完全相同的进程,这个新产生的进程称为子进程。一个进程调用fork()函数后,系统先给新的进程分配资源,例如存储数据和代码的空间。然...
  • ww1473345713
  • ww1473345713
  • 2016年06月19日 00:13
  • 5630

深入理解Linux的fork函数

一、问题引入     工作期间,某系统设计师抛出如下一个问题,下面的代码,输出几个“-”?: [cpp] view plaincopyprint? /*******************...
  • thomashtq
  • thomashtq
  • 2015年04月14日 08:08
  • 1145

浅谈 Fork/Join

fork/join的java7新增加的功能,可以把它理解成一个并发框架。我们通过fork/join能将一个可分解的大任务,分解成多个子任务同步执行,执行完毕后,在将各子任务的结果进行合并,得到最终的结...
  • cyantide
  • cyantide
  • 2016年03月19日 22:06
  • 3117

Fork/Join框架及其性能介绍

Fork/Join框架介绍Fork/Join框架是Java 7提供了的一个用于并行执行任务的框架, 大概是怎样子的呢,就是一个把大任务分割成若干个小任务,最终把每个小任务结果汇总起来得到大任务结果的框...
  • TimHeath
  • TimHeath
  • 2017年05月07日 09:19
  • 1213

linux下fork()函数用法

fork()函数用于创建子进程。进程创建成功后,函数返回两次值,一次返回值为0,代表子进程在运行, 另一次返回子进程的ID,代表当前父进程 正在运行。例如下面的程序: /*  * Test.c ...
  • xidianliusoftware
  • xidianliusoftware
  • 2011年10月25日 15:06
  • 227

linux操作系统fork详解

一、fork入门知识      一个进程,包括代码、数据和分配给进程的资源。fork()函数通过系统调用创建一个与原来进程几乎完全相同的进程,也就是两个进程可以做完全相同的事,但如果初始参数或者...
  • sinat_35925219
  • sinat_35925219
  • 2016年08月21日 10:46
  • 687

【Linux 1.0内核源码剖析】创建进程——fork.c

Linux1.0 内核源码,fork.c,主要介绍fork系统调用部分是如何创建一个新进程的,以及拷贝父进程几乎所有资源。 unix 系统通过 fork 系统调用创建一个进程,fork.c 的主要任务...
  • yeswenqian
  • yeswenqian
  • 2015年06月04日 21:26
  • 1803

Fork/Join框架介绍

1. 什么是Fork/Join框架 Fork/Join框架是Java7提供了的一个用于并行执行任务的框架, 是一个把大任务分割成若干个小任务,最终汇总每个小任务结果后得到大任务结果的框架。 我们再...
  • lubeijing2008xu
  • lubeijing2008xu
  • 2014年01月09日 11:38
  • 15023

4.2 ForkJoin_拆分任务

ForkJoin 拆分任务 概述 本章节用一个简单示例说明如何拆分任务。示例是更新一批产品的价格。任务类继承ForkJoinTask类,并复写execute方法,在execute方法中进行任务...
  • hubin0011
  • hubin0011
  • 2014年07月09日 15:04
  • 462
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:fork()函数用法理解
举报原因:
原因补充:

(最多只允许输入30个字)