有时候,我们需要在自己的程序(进程)中启动另一个程序(进程)来帮助我们完成一些工作,那么我们需要怎么才能在自己的进程中启动其他的进程呢?在Linux中提供了不少的方法来实现这一点,下面就来介绍一个这些方法及它们之间的区别。
一、system()函数调用
system函数的原型为:
#include
int system (const char *string);
它的作用是,运行以字符串参数的形式传递给它的命令并等待该命令的完成。命令的执行情况就如同在shell中执行命令:sh -c string。如果无法启动shell来运行这个命令,system()函数返回错误代码127;如果是其他错误,则返回-1。否则,system()函数将返回该命令的退出码。
注意:system()函数调用用一个shell来启动想要执行的程序,所以可以把这个程序放到后台中执行,这里system()函数调用会立即返回。
可以先运行下面的例子,源文件为new_ps_system.c,代码如下:
#include
#include
int main()
{
printf("Running ps with system");
// ps进程结束后才返回,才能继续执行下面的代码
system("ps au");// 1
printf("ps Done");
exit(0);
}
该程序调用ps程序打印所有与本用户有关的进程,最后才打印ps Done。运行结果如下:

如果把注释1的语句改为:system("ps au &");则system函数立即返回,不用等待ps进程结束即可执行下面的代码。所以你看到的输出,ps Done可能并不是出现在最后一行,而是在中间。
一般来说,使用system()函数不是启动其他进程的理想手段,因为它必须用一个shell来启动需要的程序,即在启动程序之前需要先启动一个shell,而且对shell的环境的依赖也很大,因此使用system()函数的效率不高。
二、替换进程映像——使用exec系列函数
exec系列函数由一组相关的函数组成,它们在进程的启动方式和程序参数的表达方式上各有不同。但是exec系列函数都有一个共同的工作方式,就是把当前进程替换为一个新进程,也就是说你可以使用exec函数将程序的执行从一个程序切换到另一个程序,在新的程序启动后,原来的程序就不再执行了,新进程由path或file参数指定。exec函数比system函数更有效。
exec系列函数的类型为:
#include
char **environ;
int execl (const char *path, const char *arg0, ..., (char*)0);
int execlp(const char *file, const char *arg0, ..., (char*)0);
int execle(const char *path, const char *arg0, ..., (char*)0, char *const envp[]);
int execv (const char *path, char *const argv[]);
int execvp(cosnt char *file, char *const argv[]);
int execve(const char *path, char *const argv[], char *const envp[]);
这类函数可以分为两大类,execl、execlp和execle的参数是可变的,以一个空指针结束,而execv、execvp和execve的第二个参数是一个字符串数组,在调用新进程时,argv作为新进程的main函数的参数。而envp可作为新进程的环境变量,传递给新的进程,从而变量它可用的环境变量。
承接上一个例子,如果想用exec系统函数来启动ps进程,则这6个不同的函数的调用语句为:
注:arg0为程序的名字,所以在这个例子中全为ps。
char *const ps_envp[] = {"PATH=/bin:usr/bin