进程的启动方式:
(1)通过脚本启动的方式,即内核控制启动一般放在init.rc里面。
(2)fork子进程后通过exec函数族启动新的进程
(3)采用系统的system调用来实现
在之前讲解的fork子进程的基础上通过exec函数族来看如何实现新进程的启动。
1、首先讲解一下exec()函数族
exec( )函数族
下面我们来看看一个进程如何来启动另一个程序的执行。在Linux中要使用exec函数族。系统调用execve()对当前进程进行替换,替换者为一个指定的程序,其参数包括文件名(filename)、参数列表(argv)以及环境变量(envp)。exec函数族当然不止一个,但它们大致相同,在 Linux中,它们分别是:execl,execlp,execle,execv,execve和execvp,下面我只以execlp为例,其它函数究竟与execlp有何区别,请通过manexec命令来了解它们的具体情况。
一个进程一旦调用exec类函数,它本身就"死亡"了,系统把代码段替换成新的程序的代码,废弃原有的数据段和堆栈段,并为新程序分配新的数据段与堆栈段,唯一留下的,就是进程号,也就是说,对系统而言,还是同一个进程,不过已经是另一个程序了。(不过exec类函数中有的还允许继承环境变量之类的信息。)
那么如果我的程序想启动另一程序的执行但自己仍想继续运行的话,怎么办呢?那就是结合fork与exec的使用。下面一段代码显示如何启动运行其它程序:
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#define FILE_PATH1 "../test/timeradd/bin/test"
char command[256];
void main()
{
int rtn; /*子进程的返回数值*/
/* 从终端读取要执行的命令 */
snprintf(command, "%s", FILE_PATH1, 256);
if ( fork() == 0 )
{/* 子进程执行此命令 */
execlp( command, NULL );
/* 如果exec函数返回,表明没有正常执行命令,打印错误信息*/
perror( command );
exit( errno );
}
else
{/* 父进程, 等待子进程结束,并打印子进程的返回值 */
wait ( &rtn );
printf( " child process return %d\n", rtn );
}
}
输出结果为:
main,43, i = 155242504
child process return 5888
第一行的输出时我的子进程调用进程test的输出,第二行输出是父进程的输出。
在这里有个wait函数,这个有是干什么用的呢?
2、wait()
函数原型:pid_t wait (int* status);
作用简而言之:wait会暂停当前的进程,等待子进程结束。
引用一篇博文里的话:
“进程一旦调用了wait,就立即阻塞自己,由wait自动分析是否当前进程的某个子进程已经退出,如果让它找到了这样一个已经变成僵尸的子进程,
wait就会收集这个子进程的信息,并把它彻底销毁后返回;如果没有找到这样一个子进程,wait就会一直阻塞在这里,直到有一个出现为止。”
所以上面的结果不难分析出,因为有wait函数的存在,先执行子进程的test,输出第一行,然后子进程自动退出后,wait函数停止阻塞,而输出第二行。
既然wait会一直等子进程的信息,那如果子进程一直不退出呢?是否父进程就一直在等呢?我的疑问当然都是成立的了。然子进程的test函数死循环中。父进程的打印永远也出来不了了。
所以这种用法一定要记住子进程做一件事之后马上要退出父进程才不会阻塞。
3、system函数
这个系统函数很多初学者都非常喜欢用的。研究一下C库函数system的源码会发现,system函数先调用fork,然后再调用exec()来执行用户的登陆shell,通过它来查找可执行文件的命令并执行参数。最后使用wait()来等待子进程结束。所以上面讲的简单代码实现就是一个system的类似原型。
小结:之前讲过的vfork和fork的区别明确指出,fork会拷贝父进程的数据和代码段,而vfork是和父进程共享数据段的。而fork和exec的配合调用的方式,因为是独立启动一个新的进程,所以无须重新复制父进程的数据段和代码段的,考虑到资源的浪费尽量使用vfork和exec进行组合的方式。同时由于system是用fork来实现的,所以在安全和效率方面稍有缺陷,尽量避免使用。