一起talk C栗子吧(第七十九回:C语言实例--父进程与子进程)
各位看官们,大家好,上一回中咱们说的是创建进程的例子,这一回咱们说的例子是:父进程与子进程。闲话休提,言归正转。让我们一起talk C栗子吧!
我们在上一回中介绍了如何使用fork函数创建一个新的进程。这一回中,我们在此基础上介绍父进程与子进程。也可以看作是对fork函数的进一步介绍。
什么是父进程和子进程呢?我们举例子来说明:如果进程A在运行的过程中创建了进程B,那么进程A就是父进程,进程B就是进程A的子进程。这时台下有看官提问了:“既然有父进程了,有没有母进程呢?“我的回答是没有。程序的世界属于父系社会。大家也许听说过C语言之父,C++之父,有没有人听说过C语言之母呢?或者C++之母?哈哈。台下一阵大笑。
接下来,我们通过具体的代码来说明父进程与子进程。下面是具体的代码,请大家参考:
#include #include int main() { pid_t pid; pid = fork(); sleep(5); if(pid > 0) { printf("PID: %d -> Father Process is running \n",getpid()); } else if(pid == 0) { printf("PID: %d -> Son Process is running \n",getpid()); } else { printf("Create process failed \n"); return 1; } return 0; }
在上面的代码中,我们使用fork函数创建了一个新进程,该新进程就是一个子进程。大家都知道,子进程和父进程共享相同的资源,比如代码,数据等。因此,在上面的代码中,main函数中的代码既能被父进程运行,也能被子进程运行,也就是说父进程和子进程执行相同的代码。如果真是这样的话,那么父进程和子进程运行时会输出相同的结果。
我们编译并且运行上面的程序,得到以下结果:
$ ./s //运行编译后的程序 PID: 3212 -> Father Process is running //父进程运行时输出的结果 PID: 3213 -> Son Process is running //子进程运行时输出的结果
从上面的程序运行结果可以看到,父进程和子进程运行的结果完全不同,这和我们上面分析的结果不一致呀。是程序运行出错了呢,还是我们刚才的分析不正确?
我们检查一下系统中的进程。首先,重新打开一个终端,并且在终端中使用ps命令查看当前系统中的进程,运行结果如下:
$ ps x //输入ps x命令并且执行 PID TTY STAT TIME COMMAND 1504 ? Ssl 0:00 cinnamon-session --session cinnamon ... //与系统相关的进程暂时忽略 3134 pts/2 S+ 0:01 vim //我们在使用的vim进程 3147 pts/3 Ss 0:00 bash 3190 pts/4 Ss 0:00 bash 3212 pts/3 S+ 0:00 ./s //从PID看,这是父进程 3213 pts/3 S+ 0:00 ./s //从PID看,这是子进程 3215 pts/4 R+ 0:00 ps x //使用ps命令查看进程
通过上面的结果,我们可以看出,系统中确实运行着两个进程,说明程序运行是正确的。难道是我们刚才的分析不正确?其实,我们刚才的分析和程序的运行结果都没有问题。之所以有这样的结果,是因为fork函数在不同的进程中,会产生不同的结果。
我们在刚才的分析中说:父进程和子进程执行相同的代码,更加具体点就是说,父进程和子进程都执行main函数中的代码。不过,当父进程运行到fork函数哪一行时,它会创建一个新进程,并且返回新进程的PID,这时的PID是一个大于0的正整数,因此,它只会运行下面这段代码:
if(pid > 0) { printf("PID: %d -> Father Process is running \n",getpid()); }
当子进程运行到fork函数哪一行时,它不会创建一个新进程,fork函数返回0,这时的PID等于0,因此,它会运行下面这段代码:
else if(pid == 0) { printf("PID: %d -> Son Process is running \n",getpid()); }
看官们,现在再回头看看程序的运行结果,是不是有种豁然开朗的感觉?
看官们,正文中就不写代码了,详细的代码放到了我的资源中,大家可以点击这里下载使用。
各位看官,关于父进程与子进程的例子咱们就说到这里。欲知后面还有什么例子,且听下回分解。
推荐阅读
-
一起talk C栗子吧(第一百三十九回:C语言实例--文件操作:基于文件描述符二)
-
一起talk C栗子吧(第七十九回:C语言实例--父进程与子进程)
-
一起talk C栗子吧(第八十一回:C语言实例--进程停止)
-
一起talk C栗子吧(第一百零一回:C语言实例--使用信号量进行进程间同步与互斥二)
-
一起talk C栗子吧(第一百零六回:C语言实例--生产者与消费者问题二)
-
一起talk C栗子吧(第一百零二回:C语言实例--使用信号量进行进程间同步与互斥三)
-
一起talk C栗子吧(第八十六回:C语言实例--使用信号进行进程间通信三)
-
一起talk C栗子吧(第九十七回:C语言实例--使用消息队列进行进程间通信一)
-
一起talk C栗子吧(第九十八回:C语言实例--使用消息队列进行进程间通信二)
-
一起talk C栗子吧(第一百六十九回:C语言实例--再谈内置宏)