程序替换
假如操作系统正在执行某一个程序,我们可以利用程序替换函数指定一个新的程序,让操作系统去执行我们新指定的程序。也就是这样一种情形下,我们fork一个进程,如果fork成功,子进程会和父进程执行相同的代码,而我们创建子进程是希望子进程执行指定的操作,所以需要执行exec族函数。
程序替换函数
程序替换函数是一族函数,可以通过man命令进行查看。
其中有六种以exec开头的函数,统称exec函数:(功能相同,参数不同)
#include <unistd.h>
int execl(const char *path, const char *arg, ...); //库函数
//参数:路径,操作,以NULL结尾
int execlp(const char *file, const char *arg, ...); //库函数
//参数:文件名,操作,NULL
int execle(const char *path, const char *arg, ..., char *const envp[]); //库函数
//参数:路径,操作,环境变量(空指针结尾)
int execv(const char *path, char *const argv[]); //库函数
//参数:路径,argv[](空指针结尾)
int execvp(const char *file, char *const argv[]); //库函数
//参数:文件名,argv[](空指针结尾)
int execve(const char *path, char *const argv[], char *const envp[]); //系统调用
//最标准的系统调用,参数:路径,argv,环境变量(空指针结尾)
函数解释:
- 这些函数如果调⽤用成功则加载新的程序从启动代码开始执行,不再返回
- 如果调⽤用出错则返回-1,
- 所以exec函数只有出错的返回值⽽而没有成功的返回值。exec函数族的特点是谁调用他他就替换谁,只要exec函数调用成功,后续代码全部失效。
分类说明:
不带字母p (表示path)的exec函数,第一个参数必须是程序的相对路径或绝对路径,例如”/bin/ls”或”./a.out”,而不能是”ls”或”a.out”对于带字母p的函数:如果参数中包含/,则将其视为路径名。否则视为不带路径的程序名,在PATH环境变量的目录列表中搜索这个程序。
带有字母l( 表示list)的exec函数,要求将新程序的每个命令行参数都当作一个参数传给它,命令行参数的个数是可变的,因此函数原型中有…,…中的最后一个可变参数应该是NULL, 起sentinel的作用。
带有字母v( 表示vector)的函数,则应该先构造一个指向各参数的指针数组,然后将该数组的首地址当作参数传给它,数组中的最后⼀个指针也应该是NULL,就像main函数的argv参数或者环境变量表一样。
对于以e (表示environment)结尾的exec函数,构造一份新的环境变量表传给它,构造的环境变量不会继承原有的,其他exec函数仍使用当前的环境变量表执行新程序。
#include <unistd.h>
int main(){
int ret = execl("/bin/ls", "/bin/ls", "-l", "-t", "-r", NULL);
ret = execlp("ls", "ls", "-l", "-t", "-r", NULL);
char* env[] = {"AAA=hello", NULL};
ret = execle("./hello", "./hello", NULL, env);
char* const argv1[] = {"/bin/ls", "-l", "-t", "-r", NULL};
ret = execv("/bin/ls", argv1);
char* const argv2[] = {"ls", "-l", "-t", "-r", NULL};
ret = execvp("ls", argv2);
char* const argv3[] = {"./hello", NULL};
ret = execve("./hello", argv3, env);
}