exec、execvp、execl、execv 之间的区别?

Difference between exec, execvp, execl, execv?

我正在编写代表新 shell 到 Linux 的代码。我要支持的命令之一是 运行ning 一个进程 例如,如果我得到以下行

command [arguments]

然后我想 运行 command 作为一个进程,直到它完成 运行ning 进程。

这样做我知道我需要使用 fork() 来获取子进程并获取它的 PID,我的问题是我不知道 :[=19= 之间有什么区别]

execexecvpexeclexecv ... 我不知道该使用哪一个以及为什么。

我当前的代码:

void External_Process(char *arguments[MAX_ARG], char* command)
{
    int pID;
    switch(pID = fork())
    {
    case -1:
        perror("fork failed");
        break;
    case 0 :
        setpgrp();

        //execv(command, arguments);
        //execvp(command, arguments);
        //execl("/bin/bash", "/bin/bash","-c",command,NULL);

        printf("smash error: > bad command %s\n" , command);
        exit(-1) ;

        break;

    default:
        return ;
    }
}

谢谢!

尝试阅读手册:https://linux.die.net/man/3/execv

摘录:

描述 exec() 系列函数用新的过程映像替换当前过程映像。本手册页中描述的函数是 execve(2) 的前端。 (有关替换当前进程映像的更多详细信息,请参阅 execve(2) 的手册页。)

这些函数的初始参数是要执行的文件的名称。

execl()、execlp() 和execle() 函数中的const char *arg 和后续省略号可以认为是arg0、arg1、...、argn。它们一起描述了一个或多个指向空终止字符串的指针的列表,这些字符串表示可用于已执行程序的参数列表。按照惯例,第一个参数应该指向与正在执行的文件关联的文件名。参数列表必须由 NULL 指针终止,并且由于这些是可变参数函数,因此必须将此指针转换为 (char *) NULL。

execv()、execvp() 和 execvpe() 函数提供了一个指针数组,这些指针指向以 null 结尾的字符串,表示新程序可用的参数列表。按照惯例,第一个参数应该指向与正在执行的文件关联的文件名。指针数组必须以 NULL 指针结尾。

execle() 和 execvpe() 函数允许调用者通过参数 envp 指定执行程序的环境。 envp 参数是指向以 null 结尾的字符串的指针数组,必须以 NULL 指针结尾。其他函数从调用进程中的外部变量 environ 获取新进程映像的环境。

execlp() 和 execvp() 的特殊语义

如果指定的文件名不包含斜杠 (/) 字符,execlp()、execvp() 和 execvpe() 函数会重复 shell 搜索可执行文件的操作。该文件在 PATH 环境变量中指定的以冒号分隔的目录路径名列表中查找。如果未定义此变量,则路径列表默认为当前目录,后跟 confstr(_CS_PATH) 返回的目录列表。 (此 confstr(3) 通常调用 returns 值“/bin:/usr/bin”。)

在 C 中,您可以使用 'system' 命令。 这将执行您作为函数参数输入的内容。

这是一个例子:

system("ls -l");

如果您想获取输出,您可以重定向到其他来源。

总结: 在你的情况下我建议使用 execvp.

要找出 exec* 函数之间的区别,您应该阅读文档:
https://linux.die.net/man/3/exec
https://linux.die.net/man/2/execve

execl*execv*的区别在于参数传递。 execl* 需要一个参数列表,而 execv* 需要一个参数向量。
如果您在编译时知道所有参数,则参数列表很有用。在您的情况下,参数将由用户输入,您必须在 运行 时构造一个参数向量,因此您应该使用 execv* 函数之一。

后缀为p的函数使用PATH环境变量来查找程序(例如"ls"),没有这个你必须指定完整路径(绝对或相对)到当前目录,例如 "/bin/ls")。使用 PATH 是 shell 的通常做法,因此这似乎是您的正确选择。

后缀为e的函数允许指定进程的环境。为简单起见,我不会在您的情况下使用它。

由此得出结论:execvp

当然你也可以使用 system(而不是 fork/exec*/wait*),正如 vladxjohn 的回答中提到的,但在这种情况下你只会使用 shell 来解释您的命令,而不是实现基本的 shell.