首先,我们来回忆一下父进程与子进程,前几节讲了如何创建子进程,像这样的,pid_t id = fork(); 这样我们就创建好了一个子进程,然而fork()函数的返回值是什么呢?这里要记住:子进程返回0,父进程返回子进程的pid,如果创建失败的话就返回-1.由于是父进程创建的子进程,那么子进程就继承自父进程。比如,子进程继承了父进程的数据空间,堆和栈的副本。但是,父子进程是不是就共享同一片地址空间呢?答案是否定的。这就引出了我们前几节讲的虚拟地址的内容。回忆一下:
它们就是父子进程所对应的地址空间,虽然它们的虚拟地址是一样的,但是它们的物理地址却是不一样的。(这幅图的详细介绍见本博客《linux之地址空间》)
父子进程不共享存储空间,只共享代码段。
下面看一个例子:
想一想这个代码运行的结果是什么呢?
很多人看了这个代码之后以为它会输出4条语句,但其实不然。我们来分析一下其中的奥秘吧,嘿嘿。。。
首先,父进程创建子进程,则父子进程各打印一条自己的pid,打印完之后i++;i++;这时父进程又重新创建子进程,上一级的子进程又进入到父进程,从而又创建子进程,这样i=1时相当于父子进程各创建了2个进程,即第二级创建了4个进程,所以进程数= 2+4=6.
结果如下哦:
这个图是不是有点像二叉树呢?那么当i = 10的时候呢?
经过上述的分析推理得知它的结果是这样的:2+2^2+2^3+……+2^10=2046
当i = n时,公式为:2+2^2+2^3+……+2^n = 2*(1-2^n)/(1-2);