`exec` 系列函数用于在当前进程中执行一个新程序,这将用新程序的代码和数据替换当前进程的映像。`exec` 函数族包括以下几种:
1. `execl`
2. `execle`
3. `execlp`
4. `execvp`
5. `execv`
6. `execve`
每个函数的作用是相似的,但它们在如何指定程序路径和参数上有所不同。
1. `execl`
`execl` 函数通过逐个传递参数来执行新程序,参数列表以 `NULL` 结束。
int execl(const char *path, const char *arg0, ..., (char *) NULL);
```
- `path`: 要执行的程序的路径。
- `arg0`, `...`: 程序的参数列表,第一个参数 `arg0` 通常是程序名。参数列表以 `NULL` 结束。
2. `execle`
`execle` 函数类似于 `execl`,但它允许指定自定义的环境变量。
int execle(const char *path, const char *arg0, ..., (char *) NULL, char * const envp[]);
- `envp`: 环境变量数组,最后一个元素必须是 `NULL`。
3. `execlp`
`execlp` 函数在执行程序时会在 `PATH` 环境变量指定的目录中查找程序。
int execlp(const char *file, const char *arg0, ..., (char *) NULL);
- `file`: 要执行的程序的名称。`execlp` 会在 `PATH` 环境变量指定的路径中查找该程序。
4. `execvp`
`execvp` 函数类似于 `execlp`,但它接受一个参数数组。
int execvp(const char *file, char *const argv[]);
- `file`: 要执行的程序的名称。
- `argv`: 参数数组,第一个元素通常是程序名,数组以 `NULL` 结束。
5. `execv`
`execv` 函数通过参数数组来执行新程序。
int execv(const char *path, char *const argv[]);
- `path`: 要执行的程序的路径。
- `argv`: 参数数组,第一个元素通常是程序名,数组以 `NULL` 结束。
6. `execve`
`execve` 是 `exec` 系列函数中最基础的函数,允许指定程序路径、参数数组和环境变量数组。
int execve(const char *pathname, char *const argv[], char *const envp[]);
- `pathname`: 要执行的程序的路径。
- `argv`: 参数数组,第一个元素通常是程序名,数组以 `NULL` 结束。
- `envp`: 环境变量数组,数组以 `NULL` 结束。
示例代码
以下是一个示例程序,演示了如何使用 `execvp` 执行 `ls` 命令,并打印其输出:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main() {
char *argv[] = { "ls", "-l", NULL };
// 使用 execvp 执行 `ls` 命令
execvp("ls", argv);
// 如果 execvp 成功,以下代码不会被执行
perror("execvp");
return 1;
}
错误处理
如果 `exec` 系列函数调用失败,它会返回 `-1`,并设置 `errno` 以指示错误。常见的错误包括:
- `ENOENT`: 文件或目录不存在。
- `EACCES`: 权限被拒绝。
- `ENOMEM`: 内存不足。
- `EINVAL`: 提供了无效参数。
- `EFAULT`: 提供了无效的地址。
使用 `exec` 函数的注意事项
1. 替换进程映像: `exec` 函数会替换当前进程的映像,成功执行后,原进程的代码不会再继续执行。
2. 返回值: 如果 `exec` 成功执行,新程序会替换当前进程,因此 `exec` 之后的代码不会被执行。如果调用失败,返回 `-1` 并设置 `errno`。
3. 环境变量: `execle` 和 `execve` 允许指定自定义的环境变量,而 `execvp` 和 `execlp` 会继承父进程的环境变量。